属性关键字:
- 读写权限类属性关键字:readonly 和 readwrite
- 原子性类属性关键字 : atomic(默认) 和 nonatomic
- 引用计数类属性关键字: weak、assign和unsafe_unretained(MRC使用较多)、retain和strong、copy
atomic属性关键字修饰会产生什么效果:
会保证属性赋值和获取是线程安全的。比如修饰的是一个数组,那么我们对数组进行赋值和获取是线程安全的,但是如果给数组添加、移除对象,是无法保证线程安全的。
assign和weak关键字区别:
- assign关键字可以修饰int等基本数据类型
- assign关键字修饰的时候不改变引用计数
- assign关键字修饰的对象会产生悬垂指针。对象在被释放后,assign指针仍会指向原内存地址,此时通过assign指针继续访问原对象的话,会导致内存泄露和程序异常。
- weak关键字不改变修饰对象的引用计数
- weak修饰的对象在被释放后指针会被自动置为nil
copy关键字:
1.浅拷贝:浅拷贝特点:
- 浅拷贝会增加被拷贝的引用计数
- 浅拷贝就是对内存地址的复制,没有对内存空间就行复制,让目标对象指针和原对象指针指向同一片内存空间
深拷贝特点:
- 深拷贝不会增加被拷贝对象的引用计数
- 深拷贝就是让目标对象指针和原对象指针指向两片内容相同的内存空间。
源对象类型 | 拷贝方式 | 目标对象类型 | 拷贝类型(深\浅) |
---|---|---|---|
mutable对象 | copy | 不可变类型 | 深拷贝 |
mutable对象 | mutableCopy | 可变类型 | 深拷贝 |
immutable对象 | copy | 不可变对象 | 浅拷贝 |
immutable对象 | mutableCopy | 可变类型 | 深拷贝 |
总结:
- 可变对象的
copy
和mutablecopy
都是深拷贝 - 不可变对象的
copy
是浅拷贝,mutablecopy
是深拷贝 -
copy
方法返回的都是不可变对象
相关面试题:
1.问:MRC下如何重写retain修饰的变量的setter方法?
答:
@property (nonatomic, retain) id obj;
--- 分割线 ---
- (void)setObj: (id)obj {
if (_obj != obj) {
[_obj release];
_obj = [obj retain];
}
}
2.问:下面这行代码,将一个弱引用的指针赋值给强引用的指针,可以起到强引用效果么? __strong __typeof(wself) sself = wself;
答:会的。引用计数描述的是对象而不是指针。
这句话的意思是:
sself 强引用 wself 指向的那个对象
因此对象的引用计数会增加一个。
3.问:block 内部定义了sself,会不会因此强引用了 sself?
答:不会。block 只有截获外部变量时,才会引用它。如果是内部新建一个,则没有任何问题。
4.问:如果在 block 内部没有强引用,而是通过 if 判断,是不是也可以,比如这样写:
__weak MyViewController *wself = self;
wself.completionHandler = ^(NSInteger result) {
if (wself) { // 只有当 wself 不为 nil 时,才执行以下代码
[wself.property removeObserver: wself forKeyPath:@"pathName"];
}
};
答:不可以!考虑到多线程执行,也许在判断的时候,self 还没释放,但是执行 self 里面的代码时,就刚好释放了。
5.问:那按照这个说法,block 内部强引用也没用啊。也许 block 执行以前,self 就释放了。
答:有用!如果在 block 执行以前,self 就释放了,那么 block 的引用计数降为 0,所以自己就会被释放。这样它根本就不会被执行。另外,如果执行一个为 nil 的闭包会导致崩溃。
网友评论