问题引入
因为cocos2dx底层框架是C++编写,不管业务部分使用的C++还是lua或者js开发,都是需要考虑内存问题的. 常见内存问题有:
- 程序运行中使用空指针,导致程序崩溃
- 程序中new 出来的指针没有释放(C/C++开发中特有)
- 运行内存占用越来越高
要解决这些问题,必须要理解c++的指针与内存.众所周知,C/C++申请的堆内存不会自动释放,需要由开发者主动释放.
1. malloc 和 new创建的内存
malloc 是C的函数,向操作系统申请创建一段堆内存,可能会创建失败.返回void*类型指针,内存不会初始化 new和new[]是C++ 的关键字,向操作系统申请创建一段堆内存,可能会创建失败,返回类型对象,会调用对象的构造函数.
free 是C的函数,向操作系统释放malloc开辟的内存,不会调用对象析构函数 delete 和 detele[] 是C++ 的关键字,向操作系统释放一段堆内存,会调用对象的析构函数.
注意:对于C/C++创建的堆内存,不要忘记写free delete
2. 空指针/野指针
空指针: 指向NULL ,0 ,nullptr的指针 野指针: 指针被free或者delete之后,没有设置为NULL,让人误以为这是一个合法指针,或者越界导致溢出的指针 对于空指针的错误是比较容易排查的,通常在开发阶段都会发现崩溃 野指针就比较隐蔽了,通常做法有:
- 在malloc申请内存时,将内存初始化,如memset
- 在使用delete和delete[] 后,将指针变量指向NULL
- 在使用过程中注意内存是否越界
3. 运行内存占用越来越高
内存占用越来越高通常是new开辟的内存没有得到释放,在cocos中,所有类型继承自Ref,其中定义有 unsigned int _referenceCount; 这个变量是用于计算cocos对象的引用次数,当引用计数为0时会自动释放 问题引入 因为cocos2dx底层框架是C++编写,不管业务部分使用的C++还是lua或者js开发,都是需要考虑内存问题的. 常见内存问题有:
- 程序运行中使用空指针,导致程序崩溃
- 程序中new 出来的指针没有释放(C/C++开发中特有)
- 运行内存占用越来越高
要解决这些问题,必须要理解c++的指针与内存.众所周知,C/C++申请的堆内存不会自动释放,需要由开发者主动释放.
- malloc 和 new创建的内存
malloc 是C的函数,向操作系统申请创建一段堆内存,可能会创建失败.返回void*类型指针,内存不会初始化 new和new[]是C++ 的关键字,向操作系统申请创建一段堆内存,可能会创建失败,返回类型对象,会调用对象的构造函数.
free 是C的函数,向操作系统释放malloc开辟的内存,不会调用对象析构函数 delete 和 detele[] 是C++ 的关键字,向操作系统释放一段堆内存,会调用对象的析构函数.
注意:对于C/C++创建的堆内存,不要忘记写free delete
-
空指针/野指针 空指针: 指向NULL ,0 ,nullptr的指针 野指针: 指针被free或者delete之后,没有设置为NULL,让人误以为这是一个合法指针,或者越界导致溢出的指针 对于空指针的错误是比较容易排查的,通常在开发阶段都会发现崩溃 野指针就比较隐蔽了,通常做法有: -
在malloc申请内存时,将内存初始化,如memset -
在使用delete和delete[] 后,将指针变量指向NULL -
在使用过程中注意内存是否越界 -
运行内存占用越来越高 内存占用越来越高通常是new开辟的内存没有得到释放,在cocos中,所有类型继承自Ref,其中定义有 unsigned int _referenceCount; 这个变量是用于计算cocos对象的引用次数,当引用计数为0时会自动释放
什么时候引用计数会增加
- addchild的时候,子节点引用计数会+1
- 主动调用 retain() 时引用计数+1
- 将cocos对象放入cocos定义的容器中时,引用计数+1,例如 CCArray CCVector CCList CCMap等
- cocos对象被别的cocos对象使用时引用计数+1,例如: ImageView使用的Texture,Texture对象的引用计数就会+1
什么时候引用计数会减少
- removechild /removefromParent时引用计数-1
- 主动调用release()时引用计数-1
- 将cocos对象移出cocos定义的容器中时,引用计数+1,例如 CCArray CCVector CCList CCMap等
- 调用 autorelease() 后,在下一帧引用计数-1
图片资源的管理 在程序运行中,图片资源占用内存是相当高的. 在使用图片时,cocos不是直接使用的Texture,而是使用的SpriteFrame,. cocos内部会把图片资源加入到TextureCache中缓存起来(_referenceCount+1),下次使用的时候从缓存中查找,如果没有再从文件加载为Texture,然后创建一个SpriteFrame指向这个Texture (_referenceCount+1).
SpriteFrameCache cocos加载csb/json时,其中的Sprite和ImageView对象会从SpriteFrameCache中获取使用的SpriteFrame.那么SpriteFrame是怎么来的呢,SpriteFrameCache中的SpriteFrame主要有两种来源:
-
通过Plist加载的图片 通过Plist加载的图片是一整张合图,其中有很多小图,我们可以认为每个小图都是一个SpriteFrame,一个合图就是一个Texture,SpriteFrame指向这个Texture,并记录位置,大小和旋转,路径名等. 假设一个合图有n张小图,那么Texture的_referenceCount为n+1,SpriteFrame的引用计数为1,因为有n个SpriteFrame使用到同一个Texture -
主动创建SpriteFrame添加到SpriteFrameCache中,这一点没什么好说的. 跟上面介绍的流程类似
总结 C/C++创建的内存一定要记得主动去释放 cocos对象需要注意它的引用计数,只要牢记以上规则,就能解决大多数内存问题
什么时候引用计数会增加
- addchild的时候,子节点引用计数会+1
- 主动调用 retain() 时引用计数+1
- 将cocos对象放入cocos定义的容器中时,引用计数+1,例如 CCArray CCVector CCList CCMap等
- cocos对象被别的cocos对象使用时引用计数+1,例如: ImageView使用的Texture,Texture对象的引用计数就会+1
什么时候引用计数会减少
- removechild /removefromParent时引用计数-1
- 主动调用release()时引用计数-1
- 将cocos对象移出cocos定义的容器中时,引用计数+1,例如 CCArray CCVector CCList CCMap等
- 调用 autorelease() 后,在下一帧引用计数-1
图片资源的管理
在程序运行中,图片资源占用内存是相当高的. 在使用图片时,cocos不是直接使用的Texture,而是使用的SpriteFrame,. cocos内部会把图片资源加入到TextureCache中缓存起来(_referenceCount+1),下次使用的时候从缓存中查找,如果没有再从文件加载为Texture,然后创建一个SpriteFrame指向这个Texture (_referenceCount+1).
SpriteFrameCache cocos加载csb/json时,其中的Sprite和ImageView对象会从SpriteFrameCache中获取使用的SpriteFrame.那么SpriteFrame是怎么来的呢,SpriteFrameCache中的SpriteFrame主要有两种来源:
-
通过Plist加载的图片 通过Plist加载的图片是一整张合图,其中有很多小图,我们可以认为每个小图都是一个SpriteFrame,一个合图就是一个Texture,SpriteFrame指向这个Texture,并记录位置,大小和旋转,路径名等. 假设一个合图有n张小图,那么Texture的_referenceCount为n+1,SpriteFrame的引用计数为1,因为有n个SpriteFrame使用到同一个Texture -
主动创建SpriteFrame添加到SpriteFrameCache中,这一点没什么好说的. 跟上面介绍的流程类似
总结
C/C++创建的内存一定要记得主动去释放 cocos对象需要注意它的引用计数,只要牢记以上规则,就能解决大多数内存问题
|