【setter方法内存管理之三 Objective-C语言】

前言

例如,有如下代码:
Person对象的定义如下:
@interface Person : NSObject
{
Car *_car;
}
– (void)setCar:(Car *)car;
– (Car *)car;
– (void)drive;
@end
@implementation Person
– (void)setCar:(Car *)car
{
[_car release];
_car = [car retain];
}
– (Car *)car
{
return _car;
}
– (void)drive
{
NSLog(@"走,去拉萨。。。);
[_car run];
}
– (void)dealloc
{
[_car release];
NSLog(@“人挂了。。。”);
[super dealloc];
}
@end

Car类的定义如下:
@interface Car : NSObject
{
int _speed;
}
– (void)setSpeed:(int)speed;
– (int)speed;
– (void)run;
@end
@implementation Car
– (void)setSpeed:(int)speed
{
_speed = speed;
}
– (int)speed
{
return _speed;
}
– (void)run
{
NSLog(@“时速为%d的车,在行驶”,_speed);
}
– (void)dealloc
{
NSLog(@“车挂了。。。”);
[super dealloc];
}
@end

main.h文件中:
#import <Foundtaion/Foundtaion.h>
#import “Person.h”
int main()
{
Person *p1 = [[Person alloc] init];
Car *bmw = [Car new]
bmw.speed = 100;
p1.car = bmw;
[p1 drive];
[bmw release];
bmw.speed = 200;
p1.car = bmw;
[p1 release];

return 0;
}

这时候,会报这句话发生僵尸对象错误:
– (void)setCar:(Car *)car
{
[_car release];
_car = [car retain];//这句话,会报僵尸对象错误

一、出现僵尸对象错误的原因

1)一开始,有一个Person对象p1,然后,有一个Car对象bmw,把bmw的speed属性,赋值为100,然后,把bmw作为p1对象的car属性,赋值给p1对象的car属性,这时候,怎么赋值的:
a 先把_car指向的对象release,这时候,_car为nil,release方法没有反应
b 再把bmw对象retain,再赋值给p1对象的_car属性。这时候,bmw对象的引用计数器的值为2
2)bmw对象,调用release方法,这时候,bmw对象的引用计数器的值为1
3)bmw对象的speed属性变为200
4)bmw对象,作为p1对象的_car属性,赋值给p1对象的_car属性,这时候,怎么赋值的:
a _car 指向的对象先release,_car现在指向bmw对象,bmw对象的引用计数器的值是1,release之后,bmw对象的引用计数器变为0
b bmw对象再retain,这时候,retain的了吗,retain不了,因为bmw对象已经被销毁了。
5)出现僵尸对象错误的原因:
在于, 新旧对象是同一个对象

二、解决方案

1.当发现新旧对象是同一个对象的时候,setter方法什么都不用做,只有当新旧对象不是同一个对象的时候,才release旧的,retain新的

– (void)setCar:(Car *)
{
if(_car != car)
{
[_car release];
_car = [car retain];
}
}

(_car != car)这个条件如果成立,说明什么问题啊,是不是新旧对象不是同一个对象,不是同一个对象,我才去干嘛呢,才去release旧的,retain新的。
如果是同一个对象呢,什么都不做。

2.最终完美的setter方法的写法:

– (void)setCar:(Car *)car
{
if(_car != car)
{
[_car release];
_car = [car retain];
}
}

dealloc方法最终完美版怎么写:

– (void)dealloc
{
[_car release];
[super dealloc];
}

特别注意,我们内存管理的范围,是OC对象,所以,只有属性的类型是OC对象,这个属性的setter方法,才要像上面那样写,如果属性不是OC对象类型的,setter方法直接赋值就可以了,例如Person 对象的age属性

特别注意,NSString *类型的属性,需要像上面那样写setter方法,例如Person 对象的name属性,因为NSString 是一个OC类

– (void) setName:(NSString *)name
{
if(_name != name)
{
[_name release];
_name = [ name retain];
}
}
– (void)dealloc
{
[_name release];
[super dealloc];
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

清风清晨

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值