对于初学的开发者,对于assign、retain、copy、strong、weak的用法及意义可能不是很明白,我对于这个问题也研究了很久,写篇博文,巧巧代码,让我们来瞧瞧吧!
先定义一个Student类:
#import <Foundation/Foundation.h> @interface Student : NSObject @property (nonatomic, copy) NSString *name; @end然后先是mrc下的assign声明
@property (nonatomic, assign) Student *stu1;
接下来初始化一个Student对象,并且敲入以下代码
- (void)viewDidLoad { [super viewDidLoad]; // Do any additional setup after loading the view, typically from a nib. Student *stu = [[Student alloc] init]; stu.name = @"张三"; self.stu1 = stu; NSLog(@"%p %p", &stu, &_stu1); NSLog(@"%p %p", stu,_stu1); self.stu1.name = @"李四"; NSLog(@"stu.name = %@", stu.name); NSLog(@"stu的引用计数 = %ld", [stu retainCount]); }控制台输出
(一)所以我总结,assign只是使指向stu的栈内存上的的指针,也就是stu换了一个名字,换成了stu1,就是stu和stu1的作用和意义是一样的,谁做了任何改变对应的指向栈内存的内容也会随之改变,但是栈内存的引用计数还是1没有增加。
接下来我们看看retain,改变stu1属性为以下
@property (nonatomic, retain) Student *stu1;然后重新运行程序控制台输出为:
(二)再来总结一下,retain是使指向栈内存的指针多了一个,也就是引用计数加1,并且指针stu和stu1对于栈内存的作用是一样的,也就是一扇门多了一把钥匙
接下来再看看copy的作用,同样改变stu属性为copy,但是如果是我们定义的对象,那么我们自己要实现NSCopying,这样就能调用copy,贴出代码
Student.h
#import <Foundation/Foundation.h> @interface Student : NSObject<NSCopying> @property (nonatomic, copy) NSString *name; @endStudent.m
#import "Student.h" @implementation Student - (id)copyWithZone:(NSZone *)zone { Student *stuCopy = [[[self class] allocWithZone:zone] init]; return stuCopy; } @end
viewDidLoad里的代码不变,运行程序,控制台输出
(三)可以看出,stu.name没有改变,而且stu的引用计数没用增加,为什么呢?因为copy是完全复制一段栈内存,所以copy出来的对象与原来的对象是两个不同地址的对象,所以对于stu没有影响
好了,接下来我们看看arc模式下,strong和weak的区别及用法.
先是strong,声明属性
@property (nonatomic, strong) Student *stu1;
viewDidLoad中的代码
- (void)viewDidLoad { [super viewDidLoad]; // Do any additional setup after loading the view, typically from a nib. Student *stu = [[Student alloc] init]; self.stu1 = stu; NSLog(@"%p %p", &stu, &_stu1); NSLog(@"%p %p", stu, _stu1); stu = nil; NSLog(@"将stu滞空之后"); NSLog(@"%p %p", &stu, &_stu1); NSLog(@"%p %p", stu, _stu1); }控制台输出
(四)可以看出,当将stu赋值给stu1之后,这两个指针指向的栈内存是一样的,也就是说这里和retain是一样的,但是当我们将stu滞空后,stu1指向的栈内存并没有变为空,这就是strong与retain的区别,当栈内存只要有两个或者两个以上指针指向的时候,对其中指针进行滞空操作都不会释放掉栈内存,也就是strong出来的对象,对于栈内存有控制权,还有一点,arc下默认的赋值方式就是strong
接下来,我们看看weak
其他都一样,把stu1属性改为weak,运行程序,控制台输出
(五)可以看出,当将stu赋值给stu1之后,这两个指针指向的栈内存是一样的,也就是说这里和retain是一样的,但是当我们将stu滞空后,stu1指向的栈内存变为空,这就是weak与strong的区别,也就是weak出来的对象,对于栈内存没有控制权
这里说一下,为什么不用NString测试,因为NString *str = @“dddd”,然后str = @“bbbb”,是新开辟一个栈内存来存放“bbbb”不是在原有的基础上改的。
用string做retain测试,大家会发现,无论怎么样retain,但是改变其中一个变量后,另一个不变。
版权声明:本文为博主原创文章,未经博主允许不得转载。
iOS开发-assign、retain、copy、strong、weak的区别
原文地址:http://blog.csdn.net/onethousandpiece/article/details/47689043