• Objetive-C 属性和线程安全


    一、接上一篇《nonatomic 带来的线程安全问题》,这里继续详细讨论属性各种类型与线程安全的关系

      1)影响线程安全的属性类型,nonatomic,atomic,weak

    @property (atomic, strong)   TestObject  *obj;
    @property (nonatomic, strong)   TestObject  *obj;
    @property (atomic, weak)   TestObject  *obj;
    @property (nonatomic, weak)   TestObject  *obj;
    

      上面有4种属性的定义,在遇到下面的代码的时候

        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            
            for (int i = 0; i < 10000; i++) {
                dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
    //                self.obj = [[TestObject alloc] init];
    
    
                });
            }
            
        });
    

      第1种定义不会crash,这是符合atomic的标准写法,通过setter、和getter访问器访问变量。

      第2种定义会crash,因为nonatomic定义的属性,在setter中也没有加锁,所以多线程访问之下会出现问题。

      第3种和第4种都不会crash,主要原因是,将对象赋值给weak指针的时候,都不会对对象的引用计数进行改变,

        此外weak对象添加weak引用表标记的时候,还会对该对象进行加锁,因此杜绝了多线程问题。但是多线程访问之下,最好还是atomic。

      2)如果一个对象是atomic,并且是通过下划线,实例变量访问的时候还是会crash。

        因为所谓atomic是加在getter和setter中的,我们比较一下下面属性的定义翻译成汇编之后的不同

        

    @interface ViewController ()
    
    @property (atomic, strong)   TestObject*  x;
    @property (nonatomic, strong)   TestObject*  y;
    
    @end
    

      

      对应汇编:

    注意最后调用的存储方法,一个是加锁版本,一个未加锁版本。在getter中,一个是通过方法返回的,一个是通过指针的偏移

    二、总结

      iOS中的属性和线程安全需要通过下面的方式保证:

      1) 使用atomic定义属性,同时使用getter和setter能够保证这个对象本身的引用计数线程安全问题,也就是避免多线程导致的错误释放。

      2) 使用weak定义属性,避免多线程中对引用计数的操作。

      保证了属性的原子性性访问,并不代表业务上面的线程安全问题,业务上的线程安全问题。

      业务的线程安全:

      1)加锁,NSLock,递归所,自旋锁,读写锁,内存屏障

      2)串行队列保证任务不会交替执行

      3)使用单线程+异步的模型

  • 相关阅读:
    java cp命令
    Ubuntu相关IP配置(转)
    (转)Linux操作系统下VMware的多网卡桥接转换
    Linux问题FAQ1
    hadoop运行常见问题FAQ
    hadoop运行故障问题解决1——datanode节点启动后自动关闭
    Java程序设计9——泛型
    一道灵活的css笔试题
    inherit与auto
    再谈visibility:hidden和display:none
  • 原文地址:https://www.cnblogs.com/doudouyoutang/p/9429415.html
Copyright © 2020-2023  润新知