iOS-关键字-assign weak

一,解释什么时候用assgin什么时候用weak

回答:ARC之后才有weak,weak是弱指针,当使用weak关键字修饰成员变量的时候,成员变量内部是用__weak修饰的,不会让引用计数器+1,如果指向对象被销毁,指针会自动清空,就不会报坏内存访问了

当使用assgin修饰的时候,内部是用__unsafe_unretained修饰的,不会让引用计数器+1,如果指向对象被销毁,指针不会清空   如果这时候访问对象的指针,就会有坏内存访问错误


如下图:



示例代码如下:

//  ViewController.m

#import "ViewController.h"

@interface ViewController ()

@property (nonatomic, assign 或//weak ) UIView *redView;

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
   
    UIView *view = [[UIView alloc] init];
    view.backgroundColor = [UIColor redColor];
    
//    [self.view addSubview:view];
//如果注释掉上面的代码,点击屏幕的时候是不会有view出现的,因为view是局部变量,{}之后就被销毁了, 如果使用weak修饰运行不会报错,因为view被销毁之后,指针也会自动清空,所以不会报坏内存访问
如果是assign修饰,view被销毁之后,指针不会清空, _redView.frame = CGRectMake(50, 50, 200, 200);指针访问对象的时候就会报错,因为指针没有被清空
    _redView = view;
}

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
    _redView.frame = CGRectMake(50, 50, 200, 200);
    
}

@end

补充:weak内部实现原理:


Runtime维护了一个weak表,用于存储指向某个对象的所有weak指针。weak表其实是一个hash(哈希)表,Key是所指对象的地址,Value是weak指针的地址(这个地址的值是所指对象的地址)数组。

1、初始化时:runtime会调用objc_initWeak函数,初始化一个新的weak指针指向对象的地址。

2、添加引用时:objc_initWeak函数会调用 objc_storeWeak() 函数, objc_storeWeak() 的作用是更新指针指向,创建对应的弱引用表。

3、释放时,调用clearDeallocating函数。clearDeallocating函数首先根据对象地址获取所有weak指针地址的数组,然后遍历这个数组把其中的数据设为nil,最后把这个entry从weak表中删除,最后清理对象的记录。


追问的问题一:


1.实现weak后,为什么对象释放后会自动为nil?

runtime 对注册的类, 会进行布局,对于 weak 对象会放入一个 hash 表中。 用 weak 指向的对象内存地址作为 key,当此对象的引用计数为 0 的时候会 dealloc,假如 weak 指向的对象内存地址是 a ,那么就会以 a 为键, 在这个 weak 表中搜索,找到所有以 a 为键的 weak 对象,从而设置为 nil 。


追问的问题二:


2.当weak引用指向的对象被释放时,又是如何去处理weak指针的呢?

1、调用objc_release

2、因为对象的引用计数为0,所以执行dealloc

3、在dealloc中,调用了_objc_rootDealloc函数

4、在_objc_rootDealloc中,调用了object_dispose函数

5、调用objc_destructInstance

6、最后调用objc_clear_deallocating,详细过程如下:

a. 从weak表中获取废弃对象的地址为键值的记录

b. 将包含在记录中的所有附有 weak修饰符变量的地址,赋值为   nil

c. 将weak表中该记录删除

d. 从引用计数表中删除废弃对象的地址为键值的记录



发布了4 篇原创文章 · 获赞 1 · 访问量 9万+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章