深谈UE4粒子系统对象池的惊人大坑

发布于:2022-11-28 ⋅ 阅读:(299) ⋅ 点赞:(0)

抓手

标题起得很唬人,但一点不唬人。

作者在这里告诫大家,不要使用Cascade的对象池,会有bug,并且不起作用!

建议大家都转成使用Niagara,Niagara的对象池功能齐全,并且没什么bug。

后面会具体说明原因。

同时本文会告诉大家大家具体的使用方法,以及什么情况下需要使用对象池。

本文将基于UE4.26.2的源码进行分析研究。

UE4粒子对象池的大坑

Cascade和Niagara的对象池代码原理几乎一致,但是感觉是两个人写的。(而且是负责Niagara的同学复制Cascade的对象池代码来改改)

所以Niagara有一定的优化,Cascade有坑。

具体的坑,有五个

1、Cascade的对象池没有初始化多少个实例的功能,虽然Cascade的UI上看起来可以设置,但却没有实际功能!而Niagara是有该功能的。

原因是ParticleSystem类和NiagaraSystem类都继承了父类UFXSystemAsset,都有   

uint32 PoolPrimeSize = 0; 这个变量。但ParticleSystem却没有实现相应的功能。反而NiagaraSystem在Postload方法里调用了 PostLoadPrimePools方法去做这件事。

2、从Cascade的对象池里取出的实例会重新Register Component。而Niagara则默认不重复register。这样导致Cascade粒子从对象池取出来,照样会浪费CPU的消耗,而且消耗还不低。

在我的机子11代i7 CPU上,regiseter一次Component居然需要花费2ms左右,怎么可能不卡?

 而Niagara的优化:

 Niagara还注释了为了避免Register/Unregister Component的消耗:

3、即便从Cascade的对象池取出的实例,attach到某个component上仍然调用的是SetupAttachment方法,而不是AttachToCompoent方法。但SetupComponent是构造函数里调用才生效的,所以从Cascade对象池取出的无法正确attach到component上!

4、调用SpawnEmitterAtLocation方法去Spawn粒子,从Cascade的对象池取出的实例无法正确设置它的transform!因为它调用的是SetRelativeLocation_Direct和SetRelativeRotation_Direct,该方法只能在初始化的时候调用才能起效,因此Niagara改成了SetWorldLocationAndRotation,该方法会后续用tick update的办法去更新位置和rotation。

5、Niagara有个小坑,就是AnimNotify_PlayNiagaraEffect是不开启对象池的,如果不想改引擎代码,就只能重载该方法才能使用了。同理,Cascade的AnimNotify_PlayParticleEffect也是。

Cascade的对象池源码在:

UGameplayStatics.cpp和WorldPSCPool.cpp

Niagara的对象池源码在:

NiagaraFunctionLibrary.cpp和NiagaracomponentPool.cpp

如果细看的话,这两份代码不能说是毫无关系,只能说逻辑几乎一模一样。

如果想用Cascade的对象池,建议按照Niagara的做法把Cascade的bug给修了。具体查看上面的每个对比的点。

具体使用方法(仅介绍Niagara)

初始化对象池:

打开NiagaraSystem,选择System,然后在Selection面板的System Properties ,Performance里的Pool Prime Size,填入大于0的数字:

但Pool Prime Size大于Max Pool Size,它也只会按Max Pool Size的数量来初始化,所以可以适当调整Max Pool Size数量大于或等于Pool Prime Size数量。

调用代码去Spawn时:

UNiagaraFunctionLibrary::SpawnSystemAttached或SpawnSystemAtLocation方法,在参数里传入ENCPoolMethod::AutoRelease或者ManualRelease才能使用对象池。

  • AutoRelease:在lifetime到时间后自动放回池子里。
  • ManualRelease:意味着后续需要你自己手动调用ReleaseToPool方法才能把特效放回池子里。

蓝图里Spawn:

 Pooling Method选择AutoRelease或者ManualRelease。

为什么需要使用对象池

        因为UE的粒子在Spawn的时候,ActorComponent需要注册Component,这个时候就会对CPU产生比较大的消耗。

        而使用对象池的优化原理就是,先初始化一定数量的实例在池子里,然后需要的时候再去池子里取,然后在激活使用。

        什么时候需要使用对象池呢?就是同一种粒子特效需要高频播放的时候,或者同时大量播放的时候,那么对象池是对于CPU消耗来说,是个必备良药。代价只是多占了那么点内存嘛,初始化加载时间会多一点点嘛(只要数量不会太夸张,用到成千上万的数量)。

总结:

        对于一个这么大的引擎来说,有这样的bug的确是让人意料之外,但UE也希望大家都使用Niagara,Cascade也在被慢慢遗弃,截止到4.27,Cascade对象池的问题依然存在。而UE5则默认是关闭Cascade的。

        所以本文作者在这里也呼吁大家都直接转为使用Niagara,虽然4.26的Niagara确实还有一些不好用的bug,但如果熟悉了,也是能避免的。并且Niagara更自由了。如果完全不使用对象池,则继续使用Cascade粒子是没问题的。

参考:

Unreal引擎4.26.2源码。