美文网首页iOS开发_其他
OC所有权修饰符与属性关键字

OC所有权修饰符与属性关键字

作者: Fsn_soul | 来源:发表于2016-04-19 11:25 被阅读398次

    ARC有效时,id类型和对象类型必须附加所有权修饰符,所有权修饰符一共4种:

    1. __strong修饰符.(强引用会持有对象)
    2. __weak修饰符.(弱引用不会持有对象)
    3. __unsafe_unretained修饰符
    4. __autoreleasing修饰符

    其中__strong修饰符是id类型和对象类型默认的所有权修饰符.
    使用__strong,__weak,__autoreleasing修饰符的自动变量,会被初始化为nil.
    何为内存泄露?
    内存泄露指的是应当废弃的对象在超出其生存周期后继续存在.
    仅使用__strong修饰符,会导致循环引用的发生.循环引用有两种:

    1. 相互循环引用,即两个对象间的相互强引用.
    2. 自引用导致的循环引用.

    __weak修饰符
    适用范围:只能修饰对象类型,不能用于基本数据类型.
    __weak修饰符优点:
    1.可以避免循环引用导致的内存泄露.因为弱引用不会持有对象.
    2.在持有某对象的弱引用时,若该对象被废弃,则此弱引用将自动失效且指针变量将被赋值为nil.
    __weak修饰符只能用于iOS5及以上,在iOS4只能使用__unsafe_unretained代替.不过现在已经到了最低适配版本iOS7了,所以可以放心使用__weak修饰符.
    __weak修饰符缺点:
    1.正因为附有__weak修饰符的指针变量,当它指向的对象被销毁时,系统会将该对象所有的__weak指针都置为nil,所以效率相比__unsafe_unretained修饰符要低.有时候优点也是缺点.就看你如何掌握好"度".
    2.不能用于基本数据类型.

    __unsafe_unretained修饰符
    适用范围:可用于基本数据类型也可用于对象类型.
    附有__unsafe_unretained修饰符的变量也不能持有对象.
    缺点:和__weak修饰符不同的是如果带__unsafe_unretained修饰符的变量指向的对象被废弃了那么该指针变量的值不会被置为nil,依然还是以前的值.但它已经是野指针了,再次访问将会崩溃,虽然不是每次都崩.

    __autoreleasing修饰符
    ARC有效时,将对象赋值给附有__autoreleasing修饰符的指针变量等价于在ARC无效时调用对象的autorelease方法,即将对象注册到autoreleasepool.
    但是,显式地附加__autoreleasing修饰符同显式地__strong修饰符一样罕见.因为编译器会帮我们添加.
    比如使用alloc.../开头以外的方法来取得的对象是已经被注册到了autoreleasepool里的(虽然ARC下该返回的对象不一定真的注册到autoreleasepool里,这里暂且这么理解).这是由于编译器会检查方法名是否以alloc.../开始,如果不是则自动将作为返回值的对象注册到autoreleasepool.
    比如ARC有效时,下面的方法:

    + (id)array
    
    {
    
       id obj = [[NSMutableArray alloc] init];
    
       return obj;
    
    }
    

    由于没有显式的指定所有权修饰符,所以 id obj等同于id __strong obj.由于return使得对象变量超出其作用域,所以该强引用指针变量指向的对象将被释放,但该对象作为函数的返回值,编译器会自动将其注册到autoreleasepool.这里也都没有使用显示地附加__autoreleasing修饰符.
    另外一种可以不需要显示的使用__autoreleasing修饰符的情况就是:id的指针或对象的指针(也就是双重指针)在没有显示指定时会被附加上__autoreleasing修饰符.
    最常见的例子就是,获取错误NSError时.
    NSError *err = nil; Bool result = [obj performOperationWithError:&err];
    该方法的声明为:
    - (BOOL)performOperationWithError:(NSError **)error;

    上述方法声明是等同于
    - (BOOL)performOperationWithError:(NSError * __autoreleasing *)error;

    这里再说一次:作为alloc/new/copy/mutableCopy方法返回值取得的对象是自己生成并持有的,其他情况下便是取得非自己生成并持有的对象.因此,使用附有__autoreleasing修饰符的变量作为对象取得参数,与除alloc/new/copy/mutableCopy外其他方法的返回值取得对象完全一样,都会注册到autoreleasepool,并取得非自己生成并持有的对象.

    最后,赋值给对象指针时,所有权修饰符必须一致.
    因此下面的源代码会产生编译器错误:
    NSError *err = nil; NSError **p = &err;
    需要改为:
    NSError *err = nil; NSError * __strong *p = &err;
    然而下面的这种情况又是怎么回事?
    NSError *err = nil;默认是 __strong修饰符.而方法的参数声明是__autoreleasing修饰符. NSError *err = nil; Bool result = [obj performOperationWithError:&err];

    该方法的声明为:
    - (BOOL)performOperationWithError:(NSError **)error;

    实际上,是编译器自动将上述源代码做了转换变成:
    NSError __strong *error = nil; NSError __autoreleasing *tmp = error; BOOL result = [obj performOperationWithError:&tmp]; error = tmp;

    属性关键字:
    属性关键字有:
    1.assign 对应__unsafe_unretained修饰符
    2.copy 对应__strong修饰符(但是赋值的是被复制的对象) MRC就有
    3.retain 对应__strong修饰符 在ARC有效时,依然可以使用.
    4.strong 默认 对应__strong修饰符
    5.unsafe_unretained 对应__unsafe_unretained修饰符
    6.weak 对应__weak修饰符
    7.atomic 默认
    8.nonatomic
    9.readonly
    10.readwrite 默认
    需要注意的是属性的关键字需要和它的实例变量的修饰符一致(当你不使用系统帮你生成的实例变量时)

    @interface ViewController ()
    
    {
    
        __autoreleasing NSString *_ttrsr;
    
    }
    
    @property (nonatomic, strong) NSString *ttrsr;
    
    @end
    

    这样写会报错.
    需要改为__strong NSString *_ttrsr;NSString *_ttrsr;

    相关文章

      网友评论

        本文标题:OC所有权修饰符与属性关键字

        本文链接:https://www.haomeiwen.com/subject/mkaalttx.html