java 反射 热更新_Unity热更新之C#反射动态获取类属性及方法

如果我们要为发布出去的游戏更新一些功能,但又不想让用户重新下载整个游戏包,只让他下载我们更新的资源包,用assetBundle打包资源的方式是可以的,但其中有个最大的例外,那就是脚本。

虽然assetBundle支持直接将脚本或者DLL打包在内,但没有经过unity编译的话,assetBundle中的脚本是访问不到unity总工程中的脚本的,因为他们已经完全分离在了两个程序集,用C#加载程序集的方式导入DLL的话是可行的,但即便如此,想要在assetBundle脚本与unity总工程脚本之间进行通信也还是会出现问题的。

例如我从assetBundle打包的DLL中取出的一个类NewlyAdded.cs,把他add到场景某一物体之上后,想要获取到场景物体Player上的Control.cs脚本中的float变量Health,然后再调用他的方法MassHealing(float health)。

按照正常方式:

//NewlyAdded.cs

GameObject _player = GameObject.Find("Player").gameObject;

float _f = _player.GetComponent().Health;

_player.GetComponent().MassHealing(100.0f);

如果这样做的话,我的NewlyAdded.cs脚本在发布DLL的时候就无法编译通过,因为在这段代码里他找不到Control这个类,那么只有将Control.cs也放到一起打包,这样的话虽然编译通过成功发布成DLL了,但打包成assetBundle发到客户端运行的时候你会发现GetComponent()取到的是一个奇怪的类,事实上因为被分离在了两个程序集外的原因他这里所取到的是在发布NewlyAdded.cs 的DLL那个工程里的Control类,就算本地有同名类,他是不会替换的。

解决这个问题的话,可以用到C#的反射:

//NewlyAdded.cs

float _f;

//取Player上的所有脚本

MonoBehaviour[] monos = GameObject.Find("Player").GetComponents();

for (int i = 0; i < monos.Length; i++)

{

//筛选出其中的Control脚本

if (monos[i].GetType().ToString().EndsWith("Control"))

{

//获取Health字段的值

_f = (float)monos[i].GetType().GetField("Health").GetValue(monos[i]);

//执行其公有方法MassHealing

monos[i].GetType().GetMethod("MassHealing", BindingFlags.Public).

Invoke(null, new System.Object[] { 100.0f});

break;

}

}

这样的话,在编译的时候就可以完全的通过。

d24f5a9293634c0b46dd1ba75b4c19fc.png

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值