iOS-属性修饰符

作者: FlyElephant | 来源:发表于2017-10-23 22:07 被阅读365次

    iOS开发中属性修饰符基本上每天都会打交道,网上总结也很多,本文按照实际开发的角度简单介绍一下属性修饰符,属性修饰符分为四类,指定方法名,读写属性,赋值的选项和原子性操作.

    指定方法名

    指定方法名就是自定义getter和setter方法,自定义getter和setter方法之后就不会调用系统生成的方法,Bool类型属性自定义getter和setter情况比较常见.

    @property (assign, getter=isSuccess,setter=setIsSuccess:,nonatomic) BOOL success;
    
    - (BOOL)isSuccess {
        return true;
    }
    
    - (void)setIsSuccess:(BOOL)status {
        NSLog(@"设置");
    }
    

    读写属性

    读写属性修饰符只有readwrite和readonly两种修饰符,readwrite是默认修饰符,系统会生成这个属性的setter和getter方法以及下划线开头的成员变量.

    readonly只能读不能写,生成一个getter方法下划线开头的成员变量,不会创建setter方法,直接进行赋值操作系统会报错.

    参考代码:

    @property (strong, readonly,nonatomic) Person *person;
    
    @property (copy, readonly,nonatomic) NSString *name;
    

    赋值属性

    赋值属性是包含属性修饰符最多,也是最容易出现问题的地方,赋值属性修饰符包括assign,unsafe_unretained,retain,strong,weak和copy.如果不指定赋值修饰符,系统默认的修饰符是assign.

    assign:简单赋值,不更改引用计数,用于修饰基础类型的数据(NSInteger)和C语言类型数据(int,float,double,char,bool).

    unsafe_unretained:其实质等同于assign.

    copy: 拷贝传入的对象(即创建一个引用计数为1的新对象,但是内容与传入对象相同),并把新对象赋值给实例变量.

    retain: 释放旧对象,保留传入的对象,并使传入的新对象引用计数+1.

    strong: 强引用,类似于retain,两者的区别在于修饰Block是的如果使用strong修饰,与copy修饰的结果一样,如果使用retain修饰Block相当于assign.

    @property (retain, nonatomic) TestBlock retainBlock;
    

    系统会有有警告提示:

    @property (retain, nonatomic) TestBlock retainBlock;
    

    如果使用assign修饰Block,函数作用域执行完之后Block就会失效,进行调用会直接崩溃,使用retain只有一定情况下会崩溃,Block建议使用copy修饰.

    weak: 弱引用,不保留传入的值也不会增加对象的应用计数.类似于assign,但与assign不同的是,当它们指向的对象被释放后,weak会被自动置为nil,而assign则不会,所以assign会导致“野指针”的出现,weak可以避免悬空指针.delegate建议使用weak修饰.

    原子性修饰符

    原子性操作符只有atomic和nonatomic两种修饰符,系统默认的修饰符是atomic.
    atomic属性是为了保证程序在多线程情况下,编译器会自动生成一些互斥加锁代码,避免该变量的读写不同步问题.

    nonatomic属性无需考虑多线程的情况,会让编译器少生成一些互斥加锁代码,可以提高效率.

    实现nonatomic修饰的属性的getter和setter大家都轻车熟路,如果实现一个atomic修饰的属性呢?

    @property (copy, atomic) NSString *atomicName;
    

    atomic的getter/setter实现:

    @synthesize atomicName = _atomicName;
    
    - (NSString *)atomicName {
        NSString *name;
        
        @synchronized (self) {
            name = _atomicName;
        }
        
        return name;
    }
    
    - (void)setAtomicName:(NSString *)atomicName {
        @synchronized(self) {      //加锁同步
            if (![_atomicName isEqualToString:atomicName]) {
                _atomicName = atomicName;
            }
        }
    }
    

    参考资料:
    [https://stackoverflow.com/questions/3227176/error-writable-atomic-property-cannot-pair-a-synthesized-setter-getter-with-a-u]https://stackoverflow.com/questions/3227176/error-writable-atomic-property-cannot-pair-a-synthesized-setter-getter-with-a-u

    相关文章

      网友评论

        本文标题:iOS-属性修饰符

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