重开发者的劳动成果,转载的时候请务必注明出处:http://blog.csdn.net/haomengzhu/article/details/27704153
工厂方法模式
工厂方法是程序设计中一个经典的设计模式,指的是基类中只定义创建对象的接口,将实际的实现推迟到子类中。
在这里,我们将它稍加推广,泛指一切生成并返回一个对象的静态函数。
一个经典的工厂方法如同这样:
Sprite* factoryMethod() { Sprite* ret = new Sprite(); //在这里对 ret 对象进行必要的初始化操作 return ret; }
这段看起来正常的代码其实隐藏着一个问题:
工厂方法对 ret 对象的引用在函数返回时已经结束,但是它没有释放对 ret的引用,埋下了内存泄露的隐患。
但是,如果在函数返回前就执行 release(),这显然是不合适的,
因为这会触发对象的回收,再返回的对象指针就成为了错误指针。
autorelease()方法很好地解决了这个问题。
此函数结束时我们已经丧失了对 ret 的引用,为了把 ret 对象传递给接受者,需要对它进行一次 autorelease 操作,这是因为虽然我们调用了
autorelease 方法,
但是对象直到自动回收池释放之前是不会被真正释放掉的(通常 Cocos2d-x 会在每一帧之间释放一次自动回收池),调用者有足够的时间来对它进行 retain
操作以便接管 ret 对象的引用权。
因此,Cocos2d-x 的执行机制很巧妙地保证了回收池中的对象不会在使用完毕前释放。
利用autorelease()修改后的工厂方法如下:
Sprite* factoryMethod() { Sprite* ret = new Sprite(); //在这里对 ret 对象进行必要的初始化操作 ret->autorelease(); return ret; }
调用者需要在使用完毕后谨慎地释放对象;
使用工厂方法创建对象时,虽然引用计数也为 1,但是由于对象已经被放入了回收池,
因此调用者没有对该对象的引用权,除非我们人为地调用了 retain()来获取引用权,
否则,不需要主动释放对象
关于对象传值
将一个对象赋值给某一指针作为引用的时候,为了遵循内存管理的原则,
我们需要获得新对象的引用权,释放旧对象的引用权。
此时,release()和 retain()的顺序是尤为重要的。
首先来看下面一段代码:
void SomeClass::setRef(Ref* other) { this->object->release(); other->retain(); this->object = other; }
这里存在的隐患是,当 other 和 object 实际上指向同一个对象时,第一个 release()可能会触发该对象的回收,这显然不是我们想看到的局面,所以应该先执行
retain()来保证 other 对象有效,然后再释放旧对象:
void SomeClass::setRef(Ref* other) { other->retain(); this->object->release(); this->object = other; }
其他可行的解决方案也有很多:
例如使用 autorelease()方法来代替 release()方法,或在赋值前判断两个对象是否相同。
在 Google 的 Objective-C 编程规范中,推荐使用 autorelease()方法代替 release()方法。
注意,只有两种情况你才需要调用release()方法:
(1)你new一个cocos2d::Ref子类的对象,例如CCSprite,CCLayer等。
(2)你得到coccos2d::Ref子类对象的指针,然后在你的代码中调用过retain方法。
郝萌主友情提示:
指针不是你想release就能release!!
9、Cocos2dx 3.0游戏开发找小三之工厂方法模式与对象传值,布布扣,bubuko.com