objective-c的strong与copy的区别,深拷贝与浅拷贝的区别,copy与mutableCopy的区别

strong和copy的区别

问题描述

在定义一个类的property时候,为property选择strong还是copy特别注意和研究明白的,如果property是NSString或者NSArray及其子类的时候,最好选择使用copy属性修饰。为什么呢?这是为了防止赋值给它的是可变的数据,如果可变的数据发生了变化,那么该property也会发生变化。

代码示例

还是结合代码来说明这个情况

@interface Person : NSObject
@property (strong, nonatomic) NSArray *bookArray1;
@property (copy, nonatomic) NSArray *bookArray2;
@end

@implementation Person
//省略setter方法
@end

//Person调用
main(){
    NSMutableArray *books = [@[@"book1"] mutableCopy];
    Person *person = [[Person alloc] init];
    person.bookArray1 = books;
    person.bookArray2 = books;
    [books addObject:@"book2"];
    NSLog(@"bookArray1:%@",person.bookArray1);
    NSLog(@"bookArray2:%@",person.bookArray2);
}

我们看到,使用strong修饰的person.bookArray1输出是[book1,book2],而使用copy修饰的person.bookArray2输出是[book1]。这下可以看出来区别了吧。

备注:使用strong,则person.bookArray1与可变数组books指向同一块内存区域,books内容改变,导致person.bookArray1的内容改变,因为两者是同一个东西;而使用copy,person.bookArray2在赋值之前,将books内容复制,创建一个新的内存区域,所以两者不是一回事,books的改变不会导致person.bookArray2的改变。

说到底,其实就是不同的修饰符,对应不同的setter方法,
1. strong对应的setter方法,是将_property先release(_property release),然后将参数retain(property retain),最后是_property = property。
2. copy对应的setter方法,是将_property先release(_property release),然后拷贝参数内容(property copy),创建一块新的内存地址,最后_property = property。

Objective-c中对象的Copy、MutableCopy、浅拷贝、深拷贝

对象的复制就是复制一个对象作为副本,他会开辟一块新的内存(堆内存)来存储副本对象,就像复制文件一样,即源对象和副本对象是两块不同的内存区域。对象要具备复制功能,必须实现<NSCopying>协议或者<NSMutableCopying>协议,常用的可复制对象有:NSNumber、NSString、NSMutableString、NSArray、NSMutableArray、NSDictionary、NSMutableDictionary

copy:产生对象的副本是不可变的

mutableCopy:产生的对象的副本是可变的


浅拷贝和深拷贝

      浅拷贝值复制对象本身,对象里的属性、包含的对象不做复制

      深拷贝则既复制对象本身,对象的属性也会复制一份

Foundation中支持复制的类,默认是浅复制


对象的自定义拷贝

    对象拥有复制特性,须实现NSCopying,NSMutableCopying协议,实现该协议的CopyWithZone:方法或MutableCopyWithZone:方法。

浅拷贝实现

  1. -(id)copyWithZone:(NSZone *)zone{  
  2.              
  3.            Person *person = [[[self Class]allocWithZone:zone]init];  
  4.            p.name = _name;  
  5.            p.age = _age;  
  6.            return person;  
  7.        }  

深拷贝的实现
  1. -(void)copyWithZone:(NSZone *)zone{  
  2.             Person *person = [[[self Class]allocWithZone:zone]init];  
  3.             person.name = [_name copy];  
  4.             person.age = [_age copy];  
  5.             return person;  
  6.               
  7.         }  

深浅拷贝和retain之间的关系

   copy、mutableCopy和retain之间的关系

   Foundation中可复制的对象,当我们copy的是一个不可变的对象的时候,它的作用相当与retain(cocoa做的内存优化)

   当我们使用mutableCopy的时候,无论源对象是否可变,副本是可变的

    当我们copy的 是一个可变对象时,复本不可变


转自:http://blog.csdn.net/jobtong/article/details/8453927 以及 https://segmentfault.com/a/1190000002520583


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值