程序员

OC的属性和关键字总结

2019-03-08  本文已影响3人  taobingzhi

属性

@property = ivar(成员变量)+set方法+get方法 由编译器自动组成

ps. kvc和kvo对局部成员变量无效,就是因为局部成员变量没有set/get方法。

@property有两个对应的词,一个是@synthesize(默认),一个是@dynamic。

关键字

基本类型默认关键字:atomic,readwrite,assign;
OC对象默认关键字:atomic,readwrite,strong;

这里解释一下atomic为什么是线程不安全的。

atomic本质上是对对象的set/get方法加锁,当引用A的set/get方法和引用B的set/get方法同时操作时,虽然加了锁,但是引用Bget到的可能是引用Aset之后的,所以线程不安全,且消耗性能,所以建议不用atomic关键字。

weak的实现原理。

个人理解,系统内部有一个全局的CFMutableDictionary实例,来保存每个对象的weak指针列表,key是对象地址,value是CFMutableSet类型,多个该对象的weak指针。
当对象的引用计数为0时,去这个全局的字典,通过对象地址找到所有weak指针,将其置为nil。
类似kvo的实现原理, 基于runtime实现。当对象存在weak指针时,我们可以将这个实例指向一个新创建的派生类,然后修改这个派生类的release方法,在release方法中,实现上一步所构想的事情。

可变类型和不可变类型的修饰符

可变类型只能用浅拷贝(strong),如果用深拷贝(copy),初始化会报错,因为生成的是不可变类型。
不可变类型深浅拷贝都可以,copy不会受其他对象影响,但是会在setter方法中进行判断,传入的是否可变,如果是可变就分配新的内存再赋值,如果是不可变直接赋值地址。而实际开发中大量使用的是不可变的,所以使用strong可以提升提升性能(减少一次判断),但是会受其他对象影响。

@interface Person : NSObject
@property (strong, nonatomic) NSArray *bookArray1;
@property (copy, nonatomic) NSArray *bookArray2;
@end

@implementation Person
//省略setter方法
@end

//Person调用
main(){
    NSMutableArray *books = [@[@"book1"] mutableCopy];
    Person *person = [[Person alloc] init];
    person.bookArray1 = books;
    person.bookArray2 = books;
    [books addObject:@"book2"];
    NSLog(@"bookArray1:%@",person.bookArray1);
    NSLog(@"bookArray2:%@",person.bookArray2);
}

注意block和delegate的修饰符

block用copy修饰,原因是为了延长作用域。一般情况下你不需要自行调用copy或者retain一个block. 只有当你需要在block定义域以外的地方使用时才需要copy. Copy将block从内存栈区移到堆区.其实block使用copy是MRC留下来的也算是一个传统吧, 在MRC下, 如上述, 在方法中的block创建在栈区, 使用copy就能把他放到堆区, 这样在作用域外调用该block程序就不会崩溃.但在ARC下, 使用copy与strong其实都一样, 因为block的retain就是用copy来实现的, 所以block使用copy还能装装逼, 说明自己是从MRC下走过来的。

delegate用weak修饰,为了防止循环引用。

循环引用

你中有我,我中有你,等到释放的时候,我等你释放,你等我释放,造成循环引用。特别注意block,NSTimer,和delegate。

block解决循环引用

在非arc下,可以给局部变量加一个__block修饰符来弱引用,因为非arc下,__block修饰的变量不会自动retain;在arc下,由于__block修饰的变量一样会被block retain,所以需要__weak来解决循环引用的问题。
当block内部有延时操作时,需要在内部对__weak修饰弱指针再__strong强引用一下。


Better Late Than Never!
努力是为了当机会来临时不会错失机会。
共勉!

上一篇 下一篇

猜你喜欢

热点阅读