| 导语资源池化是游戏项目中最常见的一种优化方式,具体做法是当资源不需要使用时,不直接释放而是回收到池中,等待下一次使用。对于unity游戏而言,回收的方式一般是SetActive,这么做已经回收的存量的资源就不会产生额外的消耗。但因为unity的一些“坑”,导致性能问题,我们需要换一种回收策略,也有了这篇Unity Animator卡顿研究。
在某一天,我们发现某个特效资源在使用时会有卡顿,通过Profile发现了下面这个坑
http://gameweb-img.qq.com/gad/20170731/5996e3305481fbacd31c784eb5713eec.1501493536
Animator组件在Enable时,需要重建内部数据,导致了异常的卡顿
对于这类问题,我们之前的做法是不用SetActive,而是使用SetLayer到不可见层来避免卡顿。但到底效率如何呢?为了避免同样的事情发生,针对游戏中常见的几个组件做了实验,实验的组件包括Animator, Animation, ParticleSystem, MeshRenderer
(下面所有测试都在Meizu MX4 Pro上进行,Unity版本4.7.1)
我们在一个节点下创建了4个类型的gameobject,嵌套了一层同样的子节点
http://gameweb-img.qq.com/gad/20170731/7a349cc55b2534d43a54833f98378146.1501493536
每帧分别setActive和setLayer(递归子节点)100次 结果如下
http://gameweb-img.qq.com/gad/20170731/6f60984f59a61084a65188ffcc6d4d4b.1501493537
可以看到结论是,Animator的开关耗时很大,应该尽量避免
接下来尝试用setlayer解决
创建了50个英雄(用Animator的动作),用SetLayer的方式设置成Hide层(相机不渲染),CullingType是baseOnRender
Animator的Update平均有4-5ms的开销
http://gameweb-img.qq.com/gad/20170731/356ec9160ad13b96b0a166260e05f063.1501493537
这种情况下设置成AlwaysAnimate,曲线变抖了很多,平均有6-7ms开销
http://gameweb-img.qq.com/gad/20170731/69c38d8b4d77bd1a8d09584854e75672.1501493537
SetActive之后完全没有Animator.Update了
http://gameweb-img.qq.com/gad/20170731/04c17375c67a4e1bcc9167c1d1285c55.1501493540
尝试去掉这个Update的消耗
试了几种办法:尝试Speed=0没有效果;尝试设置成到手动控制模式(StartPlayback),动作能停止但Update的消耗还在。
经过几次尝试后发现可以设置Animator的enable,又做了下面的实验:
首先测试Behaviour.Enable的性能
http://gameweb-img.qq.com/gad/20170731/9ffaf5420fec54eb46acebf697819a28.1501493540
可以看到消耗比上面几种情况都要小得多
接下来拿英雄做实验
50个英雄SetActive=false的开销还是很明显的
http://gameweb-img.qq.com/gad/20170731/4eed231b5c689d63e76c8fc889db1dfe.1501493540
SetLayer的效率高多了
http://gameweb-img.qq.com/gad/20170731/f9cda82b1517990df9393f436bb69c0f.1501493541
但Animator.Update的开销还是很高
http://gameweb-img.qq.com/gad/20170731/199e469a2c91e056eac9833f25c2c55c.1501493541
enable设置false之后Animator.Update没了
http://gameweb-img.qq.com/gad/20170731/ca094f29e745011706a09822e67c6ec6.1501493541
http://gameweb-img.qq.com/gad/20170731/11cf29307196c51fbbeb8e98d9058d39.1501493541
enable设置成true之后Animator.Update又回来了,切换开销也很小
http://gameweb-img.qq.com/gad/20170731/b8676069c377b46201b798b4983aa642.1501493542
故此,终于找一个可以两全的方法。
http://gameweb-img.qq.com/gad/20170731/444ee8416b6adc19195e54d6a8f9e02f.1501493542
结论
对于Animator组件,应该尽量避免使用SetActive,改成SetLayer Behaviour.Enable=false的效率能好很多
建议: Unity的性能优化,实践很重要。Unity引擎提供API使用很简单也很给力,但内部的原理往往不得而知。为了减少使用API造成的不良影响,一方面可以通读代码,但更重要的是通过实践和Profiler,总结出更好的做法。
网友评论