使用assign、copy、strong、weak、retain的区别在于:
- 是否开辟新的内存
- 对象的引用计数是否增加
assign和weak
- assign主要用于修饰基本数据类型,进行赋值操作,不更改引用计数; 因为基本数据类型是分配在栈上的,由系统分配和释放,不会造成野指针
- weak修饰OC对象类型,weak表示指向但不拥有该对象;修饰的对象在释放后,指针地址也会被置为nil,是一种弱引用,不会造成野指针
delegate为什么要用weak修饰
在ARC环境下,为了避免循环引用,一般会将delegate属性用weak修饰,被修饰的delegate属性引用计数不会增加,代表一种弱引用,不持有关系,而且weak修饰的对象内存被释放后,指针地址也会被置为nil;在MRC下使用assign修饰,并且手动将delegate属性置为nil
strong
strong表示指向并拥有该对象,其修饰的对象引用计数会增加1。该对象只要引用计数不为0就不会被销毁,当然强行将其设为nil可以销毁它
在MRC下,使用retain,等同于ARC下的strong
Objective-C 中,基本数据类型的默认关键字是atomic, readwrite, assign;普通属性的默认关键字是atomic, readwrite, strong。
ARC下 对象何时被释放
本质: 对象的引用计数为0时,自动释放
表象: 没有强指针指向这个对象,对象就被立即回收
什么情况下叫做对象没有强指针指向:
- 指向对象的强指针被回收(比如 强指针式局部变量)
- 指向对象的强指针被赋值为nil
引用计数是作用在指针上还是实际对象的内存?
在.m文件中 如果对象没有初始化而查看引用计数会导致崩溃,初始化之后就不会了,可以得出引用计数是针对指针所指向的对象的而不是指针本身
可变变量中,使用copy修饰,是重新开辟内存,其实就是对所修饰的对象深拷贝,strong、weak、assign三者不开辟内存,指针指向原内存地址;strong指向后该对象的内存引用计数增加1,而weak、assign不会。weak、assign会在引用计数为0时值为空,并且weak会将内存值设为nil(内存被回收,对象指针销毁),而assign可能会造成其内存已不可用,但其指针仍存在,此时调用其指针对象,会崩溃。
不可变变量中,因为值本身不可被改变,copy没必要开辟一块内存存放和原来一样的值,所以内存管理系统默认都是浅拷贝。 与可变变量一样,如weak修饰的变量同样会在内存引用计数为0时变为nil
NSString为什么使用copy修饰?
- 如果赋值源是不可变字符串,则使用strong、copy修饰,结果是一样的,因为不可变字符串本身不可变,即使修改了源字符串的值,其会新开辟一块内存存储,也不会影响这个源字符串所赋值的新字符串对象。
- 如果赋值源是可变字符串,则若想修改源字符串时不影响被赋值新生成的字符串的值,就需要用copy; 因为若使用strong,指针会指向原对象,引用计数+1,这个可变的源字符串修改的话,会影响新生成的这个字符串(不论是可变还是不可变的);若使用copy,相当于深拷贝,会生成新的内存地址和指针地址,与原字符串没有关系,修改其值不影响新生成的这个字符串(copy后的对象肯定是不可变字符串)
@property (nonatomic, strong) NSString *strongStr;
NSMutableString *originStr = [NSMutableString stringWithString:@"This is Hi"];
self.strongStr = originStr;
NSLog(@"%@ %p %p", originStr, originStr, &originStr);
NSLog(@"%@ %p %p", self.strongStr, self.strongStr, &_strongStr);
[originStr appendString:@" 嗯嗯"];
NSLog(@"%@ %p %p", originStr, originStr, &originStr);
NSLog(@"%@ %p %p", self.strongStr, self.strongStr, &_strongStr);
结果:
This is Hi 0x600002c93930 0x7ff7bc382588
This is Hi 0x600002c93930 0x7fccf4818d90
This is Hi 嗯嗯 0x600002c93930 0x7ff7bc382588
This is Hi 嗯嗯 0x600002c93930 0x7fccf4818d90
@property (nonatomic, copy) NSString *copyedStr;
NSMutableString *originStr = [NSMutableString stringWithString:@"This is Hi"];
self.copyedStr = originStr;
NSLog(@"%@ %p %p", originStr, originStr, &originStr);
NSLog(@"%@ %p %p", self.copyedStr, self.copyedStr, &_copyedStr);
[originStr appendString:@" 嗯嗯"];
NSLog(@"%@ %p %p", originStr, originStr, &originStr);
NSLog(@"%@ %p %p", self.copyedStr, self.copyedStr, &_copyedStr);
结果:
This is Hi 0x600000dd42d0 0x7ff7b20c5588
This is Hi 0x6000003e4980 0x7f80c91171a8
This is Hi 嗯嗯 0x600000dd42d0 0x7ff7b20c5588
This is Hi 0x6000003e4980 0x7f80c91171a8
参考:
iOS-使用strong,copy,assign,weak 修饰变量的异同点
iOS中copy,strong,retain,weak和assign的区别