OC基础-copy & mutableCopy

按方法区分

  • copy:产生不可变对象
  • mutableCopy:产生可变对象

按拷贝深度区分

  • 浅拷贝:不产生新对象,仅仅是对象引用计数+1,即:指针拷贝,也称地址拷贝
  • 深拷贝:产生新对象,新对象和原对象互不影响,即:内容拷贝
  • 图解:
OC基础-copy & mutableCopy_第1张图片
copy_1.png

容器对象(如:NSArray)复制

  • 对于容器对象的复制,其内部元素对象始终是指针复制
  • 这样的好处:修改任一容器中的一个元素的值可以影响到其他拷贝容器中的元素的值
  • 可以用归档的方法实现了真正的元素对象拷贝
 NSArray *array = @[[NSMutableString stringWithFormat:@"1"], @"2",@"3"];
 NSMutableArray *mutableCopy = [array mutableCopy];
// 归档实现容器元素的内容拷贝
 NSArray *trueCopy = [NSKeyedUnarchiver unarchiveObjectWithData:[NSKeyedArchiver archivedDataWithRootObject:array]];
 NSMutableString *mutableStr = [array firstObject];
 [mutableStr appendString:@"abc"];
    
 NSLog(@"%@,%@,%@", array, mutableCopy, trueCopy);
  • 输出结果
2016-08-22 21:14:19.393 test[12116:860794] (
    1abc,
    2,
    3
),(
    1abc,
    2,
    3
),(
    1,
    2,
    3
)

自定义对象实现拷贝功能

实现步骤

  • 遵守协议
  • 实现下面的方法
// zone参数是系统分配的一块内存空间
- (id)copyWithZone:(NSZone *)zone;
  • 在构建对象时记得调用的是allocWithZone方法,而不是直接调用alloc方法,如果直接调用alloc方法则会浪费系统分配的内存空间
  • 由于自定义对象一般都是可变的(只有个别系统对象需要区分可变和不可变对象,如NSString 和 NSMutableString),所以通过实现协议就可以实现深拷贝,如下面的代码示例
  • 如果有需求必须区分浅拷贝深拷贝,那么可以同时实现协议,那么copyWithZone方法中实现的就是地址拷贝,仅仅是对象引用计数+1
  • 协议对应的方法如下
// zone参数是系统分配的一块内存空间
// 调用allocWithZone产生一个新对象
- (id)mutableCopyWithZone:(NSZone *)zone;

NSObject有两个方法

// 此方法内部会调用copyWithZone:方法
// 注意:如果是自定义对象调用copy功能的话必须实现协议,否则会报未实现方法的错误
- (id)copy;

// 此方法内部会调用mutableCopyWithZone:方法
// 注意点同上
- (id)mutableCopy;

代码示例

@interface Person() 

@end

@implementation Person

// 注意这里使用allocWithZone将对象拷贝这个内存空间中
- (id)copyWithZone:(NSZone *)zone
{
    // 注意这句的写法
    Person *person = [[Person allocWithZone:zone] init];
    person.age = self.age;
    person.money = self.money;
    return person;
}

@end

copy和property

  • 属性类型为NSString、NSArray使用copy修饰符这里就不再分析了
  • 重点分析下使用copy修饰符来修饰block
    • MRC,必须使用copy修饰符,否则会造成程序奔溃
      • block默认创建出来就存储在栈内存
      • 如果不采取copy策略那么程序执行完毕后block代码块就会被销毁,那么到使用此代码块时已经被销毁了,导致程序奔溃
      • 采取copy策略将block的代码块拷贝一份放到堆内存中,这样只要引用block的对象未销毁,那么这个block代码块就不会被销毁
      • 同时block放入堆中也会带来一个新的问题,self持有block,如果在block中使用self就会产生循环引用,所以在MRC和ARC,我们分别用blcokweak来修饰self来防止循环引用
    • ARC,使用strong等同于copy

总结

  • 如果没有生成新的对象, 我们称之为浅拷贝, 本质就是指针拷贝
  • 如果生成了新的对象, 我们称之为深拷贝, 本质就是会创建一个新的对象
  • 最重要的还是记住拷贝的目的:改变原来的内容不影响副本,改变副本也不影响原来的内容

面试题,以下写法有什么问题?

@property (nonatomic, copy) NSMutableString *str;
@property (nonatomic, copy) NSMutableArray *array;
  • 由于copy策略拷贝出来的是不可变对象,而属性类型是可变对象,如果调用者误将此类属性当做可变对象来使用,将导致程序崩溃
  • 正确写法应使用strong策略
@property (nonatomic, strong) NSMutableString *str;
@property (nonatomic, strong) NSMutableArray *array;

你可能感兴趣的:(OC基础-copy & mutableCopy)