• iOS中代理属性为什么要用Weak修饰?


    一.写在前面

    代理设计模式,在iOS开发过程中,是一个非常常见的设计模式,可以说用的范围非常广泛,而对初学者来讲,常常对代理的属性修饰用weak存在疑惑,因此下面就解释一下其中非常简单的道理.

    二.必要的知识补充.

    • 众所周知,OC是一门面向对象的语言,因此,对象这个词,在iOS中是个非常重要的词汇了(扯淡).开发中,创建一个对象是再常见不过的事儿了,每个对象的创建都需要在内存中分配一定的空间,简单的说,因为内存是有限的,所以一些没有必要存在的对象,我们需要及时的去将它释放,还原更多的内存空间(如果不进行释放就会造成"内存泄露").
    • 因此,在iOS中引入了引用计数(retainCount)这个词汇,规定:只要引用计数为零,对象就会被释放,注意是规定,规定就不要问为什么了.
    • 这里主要介绍strong和weak两个修饰词(不能偏离重点),如果想了解其他的修饰词(retain,copy,getter,setter...),可以Google一下,简单点来讲strong属性会使引用计数+1,而weak修饰的对象不会使引用计数改变.
    • OK了,有了上面的只是准备,应该可以理解下面的要解释的东西了.

    三.案例说明

    1. 案例介绍

      	案例中的类介绍: 案例共用到三个类,Baby类(婴儿类),LLBabyServant类(保姆类),ViewControllr类(控制器类),其中说明问题的关键是之前两个类.
      	假设,婴儿类要想做一些事情(像换衣服了,方便了之类的...)就必须具有代理的属性,因此指定了一个协议<LLBabyDelegate>,合情合理.而保姆需要成为婴儿的代理,必须遵循协议,废话不多说,上代码!
      
    2. 首先介绍Baby类-----.h文件

      // 制定协议
      @protocol LLBabyDelegate <NSObject>
      //这里可以编写代理方法(该案例中用不到所以就不写了)
      @end
      
      @interface Baby : NSObject
      /**
       *  baby的代理属性(这里用的是weak修饰,正确的做法)
       */
      @property(nonatomic, weak) id<LLBabyDelegate>delegate;
      @end
      
      

      说明:跟上面描述的一样婴儿拥有一个代理的属性.并且用修饰词Weak修饰的.

      接着粘Baby类-----.m文件

      @implementation Baby
      - (void)dealloc
      {
      	NSLog(@"Baby被销毁了");
      }
      @end
      

      说明:该方法的作用是当该类的对象被销毁时会调用-dealloc方法(在这个案例中用来观察对象是否被销毁了)

    3. 再有介绍BabyServant类-----.m文件

      //BabyServant类也就是保姆必须遵循代理协议	
      @interface BabyServant()<LLBabyDelegate>
      //并有一个需要照顾的Baby
      @property(nonatomic, strong) Baby *baby;
      
      @end
      
      @implementation LLBabyServant
      //初始化方法
      - (instancetype)init
      {
       self = [super init];
      
      	if (self) {
          // 初始化'婴儿'对象
          self.baby = [[Baby alloc]init];
          // 设置'自己(LLBabyServant)'为(Baby)类的代理
          self.baby.delegate = self;
      	}
      
       return self;
      }
      
    4. 我们介绍最后一个类 ViewController类

      @implementation ViewController
      
      - (void)viewDidLoad {
      	[super viewDidLoad];
      
       	LLBabyServant *servant = [[LLBabyServant alloc]init];
      
      	}
      
      @end
      
      

      说明: 在ViewController的方法ViewDidLoad中创建保姆对象,并且该对象作用的范围是这个方法内部.这个方法执行完成,servant对象就会被销毁了.

      上面的代码执行完成之后,运行结果如下:

       2016-02-18 15:06:41.151 代理weak案例[1805:173874] LLBabyServant	被销毁了
       2016-02-18 15:06:41.152 代理weak案例[1805:173874] Baby类被销毁了
      

      这就说明LLBabyServant对象和Baby对象在没有用处之后都会被销毁,但是如果用代理用strong修饰,而不是用weak修饰,则不会打印上面的结果!

    四.分析

    • 用weak分析图如下:

      如果weak的话,在程序运行的时候不会造成循环引用,对象都会被顺利的销毁,所以会调用婴儿类和保姆类的delloc方法,
      
    • 用strong分析如下

      如果strong的话,在程序运行的时候会造成循环引用(意思就是reatainCount不为0,只要有实线引用,计数器就+1),对象都不会的销毁,所以会调用婴儿类和保姆类不会调用delloc方法,从而造成了内存泄露的问题,
      

    这就是为什么代理属性要用weak修饰的原因了,希望能帮到你们!

  • 相关阅读:
    jQuery
    BOM与DOM操作
    剑指offer 面试题10.1:青蛙跳台阶
    剑指offer 面试题10:斐波那契数列
    剑指offer 面试题9.1:用两个队列实现栈
    剑指offer 面试题9.1:用两个队列实现栈
    剑指offer 面试题9:用两个栈实现队列
    剑指offer 面试题8:二叉树的下一个节点
    剑指offer 面试题7:重建二叉树
    剑指offer 树的基本操作:四种遍历方式
  • 原文地址:https://www.cnblogs.com/upliver/p/5198410.html
Copyright © 2020-2023  润新知