博客> 究竟什么时候用copy,什么时候用strong
究竟什么时候用copy,什么时候用strong
2018-10-19 14:57 评论:0 阅读:597 阮延超
属性 copy 拷贝

在说这个命题的答案之前,先跟大家说说深拷贝和浅拷贝;把这两点弄清楚,再去理解copy 和 strong 用法的区别就非常简单了。 深拷贝就是复制一份内容,而浅拷贝就是拷贝一份指针 光光知道这些还远远不够,在代码中实际操作的时候,我们碰到的是copy 和 mutableCopy的使用 举个例子

copy/mutableCopy NSString

_NSString string = @"美国队长3"; // 没有产生新对象 NSString copyString = [string copy]; // 产生新对象 NSMutableString *mutableCopyString = [string mutableCopy];

NSLog(@"string = %p copyString = %p mutableCopyString = %p", string, copyString, mutableCopyString);_ 结果是 20150811132929685.png

copy/mutableCopy NSMutableString

NSMutableString *string = [NSMutableString stringWithString:@"美国队长3"];

// 产生新对象 NSString copyString = [string copy]; // 产生新对象 NSMutableString mutableCopyString = [string mutableCopy];

NSLog(@"string = %p copyString = %p mutableCopyString = %p", string, copyString, mutableCopyString);

结果 20150811133013548.png 结论  深拷贝.png

其他对象NSArray、NSMutableArray 、NSDictionary、NSMutableDictionary一样适用  所有类型拷贝.png

copy 的秘密 用copy 其实本质是上是调用了对象的鼻祖NSObject里面的

  • (id)copyWithZone:(NSZone *)zone; 如果类没有实现这个方法,就会crash; _#import "HSPerson.h"

@interface HSPerson()

@end

@implementation HSPerson

  • (id)copyWithZone:(NSZone *)zone {

    return @"美国队长3"; }

@end_ _HSPerson *p = [[HSPerson alloc] init]; p.age = 20; p.height = 170.0;

HSPerson copyP = [p copy]; NSLog(@"copyP: %@", copyP); 结果是 可以看出copyWithZone重新分配新的内存空间,则: - (id)copyWithZone:(NSZone )zone { HSPerson *person = [[HSPerson allocWithZone:zone] init]; return person;

// 有些人可能下面alloc,重新初始化空间,但这方法已给你分配了zone,自己就无需再次alloc内存空间了 // HSPerson *person = [[HSPerson alloc] init]; } _

HSPerson *p = [[HSPerson alloc] init]; p.age = 20; p.height = 170.0;

HSPerson *copyP = [p copy]; NSLog(@"p = %p copyP = %p", p, copyP);

NSLog(@"age = %d height = %f", copyP.age, copyP.height); 结果是 20150811133216440.png 虽然copy了份新的对象,然而age,height值并未copy,那么需要在方法里传值 - (id)copyWithZone:(NSZone )zone { HSPerson person = [[HSPerson allocWithZone:zone] init]; person.age = self.age; person.height = self.height; // 这里self其实就要被copy的那个对象,很显然要自己赋值给新对象,所以这里可以控制copy的属性 return person; }

结果是 20150811133229314.png

NSMutableCopying 跟copy的实现并无太大差异 _- (id)mutableCopyWithZone:(NSZone )zone { HSPerson person = [[HSPerson allocWithZone:zone] init]; person.age = self.age; person.height = self.height;

return person;

}_

接下来重点来了,属性的copy跟strong的区别使用 首先说Copy _#import

@interface HSPerson : NSObject

@property (nonatomic, copy) NSString *name;

@end

NSMutableString *string = [NSMutableString stringWithFormat:@"蜘蛛侠2"];

HSPerson *person = [[HSPerson alloc] init]; person.name = string;

// 不能改变person.name的值,因为其内部copy新的对象 [string appendString:@" hans"];

NSLog(@"name = %@", person.name); _ 结果是“蜘蛛侠2”

copy实际上对name干了一件事

  • (void)setName:(NSString *)name { _name = [name copy]; }

假设name为NSMutableString,会发生什么事?实际上内部还是:

  • (void)setName:(NSMutableString *)name { _name = [name copy]; } copy出来的仍然是不可变字符!如果有人用NSMutableString的方法,就会崩溃:  20150811133407022.png

再说说strong @property (nonatomic, strong) NSString name; NSMutableString string = [NSMutableString stringWithFormat:@"蝙蝠侠"];

HSPerson *person = [[HSPerson alloc] init]; person.name = string;

// 可以改变person.name的值,因为其内部没有生成新的对象 [string appendString:@" 黑暗骑士"];

NSLog(@"name = %@", person.name);

结果是 “蝙蝠侠 黑暗骑士”

结论就是copy是给不可更改属性的,而strong给可以修改的属性的

收藏
0
sina weixin mail 回到顶部