1 PNG和ETC的加载区别
1.1 PNG
PNG是一种图片压缩格式,并不能被GPU直接识别,CPU把PNG图片读取到内存后,还需要在内存中解码,转换成GPU能识别的数据格式,然后传送给GPU渲染。
整个过程所占内存是编码数据*1+解码数据*2
,即原始PNG文件占一份,解码后的数据在内存和GPU各占一份。
2.1 ETC
ETC是一种能被GPU直接识别的压缩纹理格式,CPU把ETC压缩纹理读取到内存后,无需解码,直接传送给GPU渲染。
整个过程所占内存是压缩纹理数据*2
。
2 测试环境
- CocosCreator 2.4.x
- 构建Android包在真机上测试
- 关闭纹理缓存,即 cc.macro.CLEANUP_IMAGE_CACHE = true,使加载后的图片内存及时释放。
- 4种尺寸的图片:等于2048、小于2048、等于1024、小于1024,各使用4张不同的。
3 内存对比
3.1 加载PNG
3.1.1 长宽等于2048的图片
理论内存是16M。第一次加载完后,内存增加约32M,后面再加载了3张相同大小的不同图片,每次增加约16M。
为什么第一次会增加32M呢?在 cc.macro.CLEANUP_IMAGE_CACHE = true 的情况下,正常应该是只占16M。如果不是引擎BUG的话,推测是因为js的数据内存由gc控制,不是使用完就立即回收的。
3.1.2 长宽小于2048且大于1024的图片
4张图片的详细尺寸稍有区别,具体就不列举了,反正理论内存约12M。第一次加载完后,内存增加约24M,后面3次加载,每次增加约12M。为什么第一次是2倍实际内存即24M呢?推测原因同上。
3.1.3 长宽等于1024的图片
理论内存是4M。第一次加载完后,内存增加约8M,后面3次加载,每次增加约4M。
3.1.4 长宽小于1024的图片
理论内存是1-2M。第一次加载完后,内存增加约4M,后面3次加载,每次增加约1-2M。
3.2 加载ETC1纹理
加载ETC1的表现和PNG类似,就不再详述了,也是第一次加载会占用双倍内存,但后面就不会了。但平均来说,每次增加内存只有PNG的1/4,优势巨大。
4 总结
- OpenGL ES 2.0开始,GPU纹理支持非2的幂次方。例如,小于2048的图片比2048图片更省内存。目前所有手机都已经是OpenGL ES 2.0及以上了(2021年)。
- 尽量使用压缩纹理,ETC1的内存占用只相当于PNG的1/4,但文件体积约比PNG大1-2倍,经过zip后比PNG小。
- ETC2的质量比ETC1好,但需要OpenGL ES 3.0支持,目前(2021年)不是所有Android手机都支持,但iPhone6开始已经是OpenGL ES 3.0了,所以iOS可以考虑用ETC2。
- 图片长宽比要尽量均匀,例如,一张
120*2000
的图片,最好转换成480*500
。 - PVR纹理要求长宽是正方形且是2的幂次方,如果用PVR格式,则图片大小应该依次限制在256、512、1024、2048内,宁愿用3张1024也比用一张2048好。