草庐IT

【HybirdCLR】入门记录-Unity2021 + WebGL

xiaoLongww 2024-01-08 原文

目录


前言

之前一直有听说热更新技术,于是找点时间来研究一下热更新技术的使用。热更新的实现方式有很多种,这里笔者记录一下自己学习HybirdCLR的过程。


环境

unity2021.3.10f1c2,visual studio 2019

案例学习

先PC平台试一下

下载官方示例后,按照readme文档说的进行操作:

  1. 打开Installer,点击安装,等待安装完成
  2. HybirdCLR/Generate/All点击一下
  3. HybirdCLR/Build/Win64点击一下,生成exe
  4. 然后可以打开 hybridclr_trial-main\Release-Win64\HybridCLRTrial.exe 会看到打出 hello, HybridCLR.prefab

转为WebGL平台

  1. build setting更改到WebGL平台
  2. 因为是WebGL,按照官方文档的说法,需要勾选UseGlobal II2CPP,并且把hybridclr_trial-main\HybridCLRData\LocalIl2CppData-WindowsEditor\il2cpp,覆盖到编辑器的相似路径——unityEditor\2021.3.10f1c2\Editor\Data\il2cpp
  3. 运行菜单 HybridCLR/Generate/All 一键执行必要的生成操作
  4. Build Settings里打包游戏
  5. 运行菜单 HybridCLR/Build/BuildAssetsAndCopyToStreamingAssets 打包热更新资源及dll
  6. Assets/StreamingAssets下的所有文件复制到你刚才打包的游戏的StreamingAssets目录(如果是直接打android apk包,则再次Build即可)
  7. 运行刚刚打包成功的游戏,应该会出现Script Missing的警告,因为目前HybirdCLR还不支持unity2021的WebGL在AB包上直接挂载脚本,或许以后大佬们会支持吧。所以说要改用反射的方式去调用:
    官方-使用反射来使用热更新代码

动手做一个demo

功能

有个UGUI的text,显示热更新程序集的脚本中方法返回的字符串

基本工作流程

搭建环境

  1. 新建项目,切换为webgl
  2. 包管理器从git获取,https://gitee.com/focus-creative-games/hybridclr_unity.git
  3. 在installer中选择安装
  4. 修改playerSetting—— Api改成 .NET Framework
  5. 发现在WebGL平台,GC选项是关闭的。所以去到PC平台,关闭增量式GC(Use Incremental GC) 选项
  6. 打包一下,确认WebGL可以在本机IIS上运行
  7. HybirdCLR/Settings打开设置,勾选UseGlobal II2CPP。因为用的同一个版本的unity编辑器,所以就不用再覆盖了。然后在热更新DLLS填写Assembly-CSharp,表示将Assembly-CSharp程序集作为热更新程序集。

构建项目

  1. 官方推荐新手将Assembly-CSharp作为热更新程序集,那么笔者这里也创建一个新的程序集作为热更新的入口——Main.asmdef
  2. 新建文件夹A(任意名称)来放主入口程序集,把Main.asmdef放进来
  3. 模仿官方案例的设置,在Main.asmdef中添加对HybirdRuntime的引用
  4. 在A中新建脚本LoadDll.cs,模仿官方案例即可,就改了一小段:
  void StartGame()
    {
        LoadMetadataForAOTAssemblies();

        ass = System.Reflection.Assembly.Load(GetAssetData("Assembly-CSharp.dll"));

        var klass = ass.GetType("SaySth");
        var method = klass.GetMethod("SayHello");
        string str =  (string)method.Invoke(null, null);
        textMeshPro.text = str;

    }

当然你得在Assembly-CSharp中先创建个脚本:

using System.Collections;
using System.Collections.Generic;
using TMPro;
using UnityEngine;

public class SaySth : MonoBehaviour
{
    public static string SayHello()
    {
         string tes = "hello,world";
        //string tes = "---------hello,world";

        //Debug.Log(tes);
        return tes;
    }

}

  1. 然后把官方案例的Editor文件夹的脚本BuildAssetsCommand.cs拷过来用。因为没有使用预制体打包,所以稍微裁剪一下:
using HybridCLR.Editor.Commands;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using UnityEditor;
using UnityEngine;

namespace HybridCLR.Editor
{
    public static class BuildAssetsCommand
    {
       
        [MenuItem("HybridCLR/Build/BuildAndCopyAOTHotUpdateDllsToStreamingAssets")]
        public static void BuildAndCopyAOTHotUpdateDlls()
        {
            BuildTarget target = EditorUserBuildSettings.activeBuildTarget;

            CompileDllCommand.CompileDll(target);

            CopyAOTHotUpdateDlls(target);
        }

        public static void CopyAOTHotUpdateDlls(BuildTarget target)
        {

            CopyAOTAssembliesToStreamingAssets();
            CopyHotUpdateAssembliesToStreamingAssets();
        }

        /// <summary>
        /// 元数据dll名称,为了解决AOT泛型问题
        /// </summary>
        public static List<string> AOTMetaAssemblyNames { get; } = new List<string>()
        {
            "mscorlib.dll",
            "System.dll",
            "System.Core.dll",
        };

        /// <summary>
        /// 把AOT元数据丢到streamingasset
        /// </summary>
        public static void CopyAOTAssembliesToStreamingAssets()
        {
            var target = EditorUserBuildSettings.activeBuildTarget;
            string aotAssembliesSrcDir = SettingsUtil.GetAssembliesPostIl2CppStripDir(target);
            string aotAssembliesDstDir = Application.streamingAssetsPath;

            foreach (var dll in AOTMetaAssemblyNames)
            {
                string srcDllPath = $"{aotAssembliesSrcDir}/{dll}";
                if (!File.Exists(srcDllPath))
                {
                    Debug.LogError($"ab中添加AOT补充元数据dll:{srcDllPath} 时发生错误,文件不存在。裁剪后的AOT dll在BuildPlayer时才能生成,因此需要你先构建一次游戏App后再打包。");
                    continue;
                }
                string dllBytesPath = $"{aotAssembliesDstDir}/{dll}.bytes";
                File.Copy(srcDllPath, dllBytesPath, true);
                Debug.Log($"[CopyAOTAssembliesToStreamingAssets] copy AOT dll {srcDllPath} -> {dllBytesPath}");
            }
        }

        /// <summary>
        /// 将热更新dll改名,丢到streamingasset
        /// </summary>
        public static void CopyHotUpdateAssembliesToStreamingAssets()
        {
            var target = EditorUserBuildSettings.activeBuildTarget;

            string hotfixDllSrcDir = SettingsUtil.GetHotUpdateDllsOutputDirByTarget(target);
            string hotfixAssembliesDstDir = Application.streamingAssetsPath;
            foreach (var dll in SettingsUtil.HotUpdateAssemblyFiles)
            {
                string dllPath = $"{hotfixDllSrcDir}/{dll}";
                string dllBytesPath = $"{hotfixAssembliesDstDir}/{dll}.bytes";
                File.Copy(dllPath, dllBytesPath, true);
                Debug.Log($"[CopyHotUpdateAssembliesToStreamingAssets] copy hotfix dll {dllPath} -> {dllBytesPath}");
            }
        }


    }
}

  1. 然后可以点击HybirdCLR/Build/BuildAndCopyAOTHotUpdateDllsToStreamingAssets,将热更新程序集和3个AOT用到的元数据补充程序集改名打包到StreamingAssets中
  2. 点击Build打包即可
  3. 现在可以修改SayHello方法中的逻辑,来查看热更新的效果。修改后再次点击HybirdCLR/Build/BuildAndCopyAOTHotUpdateDllsToStreamingAssets来更新Assembly-CSharp.dll.bytes文件的内容,然后简单替换这个文件即可。可以看到笔者这里分别输出了不同的字符串,改变了文本;

补充

来回切换Assembly-CSharp.dll.bytes时,笔者发现有时候文本没有更新。查看IIS缓存配置,然后修改了IIS如下,这样切换bytes文件后,可以及时更新了。


致谢

最后要感谢一下群里大佬们的耐心教导,这个群非常的好,人多说话又好听,新手群号

428404198

有时间的话建议再看看官方文档里面对原理的描述,深刻理解一下与其他主流热更新框架的区别。

参考资料

热更新的基础知识,以及一些主流热更新框架介绍
HybridCLR/huatuo,GitHub
unity中的AOT、JIT、IL2CPP、Mono
unity程序集——类似DLL,能够把unity工程中的脚本划分到不同的程序集
HybridCLR官方文档

有关【HybirdCLR】入门记录-Unity2021 + WebGL的更多相关文章

  1. ruby - Sinatra:运行 rspec 测试时记录噪音 - 2

    Sinatra新手;我正在运行一些rspec测试,但在日志中收到了一堆不需要的噪音。如何消除日志中过多的噪音?我仔细检查了环境是否设置为:test,这意味着记录器级别应设置为WARN而不是DEBUG。spec_helper:require"./app"require"sinatra"require"rspec"require"rack/test"require"database_cleaner"require"factory_girl"set:environment,:testFactoryGirl.definition_file_paths=%w{./factories./test/

  2. ruby-on-rails - Rails 5 Active Record 记录无效错误 - 2

    我有两个Rails模型,即Invoice和Invoice_details。一个Invoice_details属于Invoice,一个Invoice有多个Invoice_details。我无法使用accepts_nested_attributes_forinInvoice通过Invoice模型保存Invoice_details。我收到以下错误:(0.2ms)BEGIN(0.2ms)ROLLBACKCompleted422UnprocessableEntityin25ms(ActiveRecord:4.0ms)ActiveRecord::RecordInvalid(Validationfa

  3. Unity 热更新技术 | (三) Lua语言基本介绍及下载安装 - 2

    ?博客主页:https://xiaoy.blog.csdn.net?本文由呆呆敲代码的小Y原创,首发于CSDN??学习专栏推荐:Unity系统学习专栏?游戏制作专栏推荐:游戏制作?Unity实战100例专栏推荐:Unity实战100例教程?欢迎点赞?收藏⭐留言?如有错误敬请指正!?未来很长,值得我们全力奔赴更美好的生活✨------------------❤️分割线❤️-------------------------

  4. FOHEART H1数据手套驱动Optitrack光学动捕双手运动(Unity3D) - 2

    本教程将在Unity3D中混合Optitrack与数据手套的数据流,在人体运动的基础上,添加双手手指部分的运动。双手手背的角度仍由Optitrack提供,数据手套提供双手手指的角度。 01  客户端软件分别安装MotiveBody与MotionVenus并校准人体与数据手套。MotiveBodyMotionVenus数据手套使用、校准流程参照:https://gitee.com/foheart_1/foheart-h1-data-summary.git02  数据转发打开MotiveBody软件的Streaming,开始向Unity3D广播数据;MotionVenus中设置->选项选择Unit

  5. unity---接入Admob - 2

    目录1.AdmobSDK下载地址2.将下载好的unityPackagesdk导入到unity里​编辑 3.解析依赖到项目中

  6. Unity 3D 制作开关门动画,旋转门制作,推拉门制作,门把手动画制作 - 2

    Unity自动旋转动画1.开门需要门把手先动,门再动2.关门需要门先动,门把手再动3.中途播放过程中不可以再次进行操作觉得太复杂?查看我的文章开关门简易进阶版效果:如果这个门可以直接打开的话,就不需要放置"门把手"如果门把手还有钥匙需要旋转,那就可以把钥匙放在门把手的"门把手",理论上是可以无限套娃的可调整参数有:角度,反向,轴向,速度运行时点击Test进行测试自己写的代码比较垃圾,命名与结构比较拉,高手轻点喷,新手有类似的需求可以拿去做参考上代码usingSystem.Collections;usingSystem.Collections.Generic;usingUnityEngine;u

  7. LC滤波器设计学习笔记(一)滤波电路入门 - 2

    目录前言滤波电路科普主要分类实际情况单位的概念常用评价参数函数型滤波器简单分析滤波电路构成低通滤波器RC低通滤波器RL低通滤波器高通滤波器RC高通滤波器RL高通滤波器部分摘自《LC滤波器设计与制作》,侵权删。前言最近需要学习放大电路和滤波电路,但是由于只在之前做音乐频谱分析仪的时候简单了解过一点点运放,所以也是相当从零开始学习了。滤波电路科普主要分类滤波器:主要是从不同频率的成分中提取出特定频率的信号。有源滤波器:由RC元件与运算放大器组成的滤波器。可滤除某一次或多次谐波,最普通易于采用的无源滤波器结构是将电感与电容串联,可对主要次谐波(3、5、7)构成低阻抗旁路。无源滤波器:无源滤波器,又称

  8. 微信小程序开发入门与实战(Behaviors使用) - 2

    @作者:SYFStrive @博客首页:HomePage📜:微信小程序📌:个人社区(欢迎大佬们加入)👉:社区链接🔗📌:觉得文章不错可以点点关注👉:专栏连接🔗💃:感谢支持,学累了可以先看小段由小胖给大家带来的街舞👉微信小程序(🔥)目录自定义组件-behaviors    1、什么是behaviors    2、behaviors的工作方式    3、创建behavior    4、导入并使用behavior    5、behavior中所有可用的节点    6、同名字段的覆盖和组合规则总结最后自定义组件-behaviors    1、什么是behaviorsbehaviors是小程序中,用于实现

  9. 【Java入门】使用Java实现文件夹的遍历 - 2

    遍历文件夹我们通常是使用递归进行操作,这种方式比较简单,也比较容易理解。本文为大家介绍另一种不使用递归的方式,由于没有使用递归,只用到了循环和集合,所以效率更高一些!一、使用递归遍历文件夹整体思路1、使用File封装初始目录,2、打印这个目录3、获取这个目录下所有的子文件和子目录的数组。4、遍历这个数组,取出每个File对象4-1、如果File是否是一个文件,打印4-2、否则就是一个目录,递归调用代码实现publicclassSearchFile{publicstaticvoidmain(String[]args){//初始目录Filedir=newFile("d:/Dev");Datebeg

  10. ES基础入门 - 2

    ES一、简介1、ElasticStackES技术栈:ElasticSearch:存数据+搜索;QL;Kibana:Web可视化平台,分析。LogStash:日志收集,Log4j:产生日志;log.info(xxx)。。。。使用场景:metrics:指标监控…2、基本概念Index(索引)动词:保存(插入)名词:类似MySQL数据库,给数据Type(类型)已废弃,以前类似MySQL的表现在用索引对数据分类Document(文档)真正要保存的一个JSON数据{name:"tcx"}二、入门实战{"name":"DESKTOP-1TSVGKG","cluster_name":"elasticsear

随机推荐