oc中copy的简单使用以及注意事项

OC中copy修饰@property

1.copy修饰字符串,可以防止外界修改内部的数据

@interfaceStudent:NSObject@property(nonatomic,strong)NSString*name;@end

NSMutableString*temp=[[NSMutableString alloc]initWithFormat:@"lwl"];Student*stu=[[Student alloc]init];stu.name=temp;[temp appendString:@" test"];NSLog(@"name = %@",stu.name);

首先我们创建了一个Student类,并声明了一个name属性,修饰符为strong。在代码二中,定义了一个变量temp可变字符串,并创建了stu实例。把temp字符串赋值给stu中的name属性。最后temp字符串追加@“test”,最后打印出来的stu的name也会追加test;

2019-01-2714:50:25.484899+0800CopyUserProperty[8486:938202]Hello,World!2019-01-2714:50:25.485079+0800CopyUserProperty[8486:938202]name=lwl testProgram endedwithexit code:0

**解决办法**

只需要将Student中的name属性修饰的strong变成copy即可

@interfaceStudent:NSObject@property(nonatomic,copy)NSString*name;@end

2019-01-2715:12:39.131612+0800CopyUserProperty[9016:1066999]Hello,World!2019-01-2715:12:39.131878+0800CopyUserProperty[9016:1066999]name=lwlProgram endedwithexit code:0

**分析原因**

如果修饰符为strong,在将temp赋值给stu的name的时候,temp指向的对象会在引用计数(retaincount)会加1,相单于两个变量同时指向了同一个对象,修改其中一个另外一个肯定会变换。如果修饰符为copy,在将temp赋值给stu的name的时候,会重新拷贝一个对象放在堆中,temp指向的对象的引用计数(retaincount)不会加1,temp和stu的name指向不同的对象。是深拷贝。

2.copy修饰block

block默认存储在栈中,栈中的block访问到外界的对象时,不会进行相应的retain操作;如果block在堆中(Block_copy宏可以将block转移到堆中),在访问外界的对象时,会进行相应的retain操作。

Student*stu=[[Student alloc]init];NSLog(@"retainCount = %lu",[stu retainCount]);void(^myBlock)(int)=^(int age){NSLog(@"num = %@",stu);};myBlock(3);Block_copy(myBlock);NSLog(@"retainCount = %lu",[stu retainCount]);

Student类新增block属性

typedefvoid(^myBlock)(void);@interfaceStudent:NSObject@property(nonatomic,assign)myBlock pBlock;@property(nonatomic,copy)NSString*name;@end

Dog*d=[[Dog alloc]init];Student*stu=[[Student alloc]init];stu.pBlock=^{NSLog(@"d = %@",d);};[d release];stu.pBlock();[stu release];

在stu对象中的block中访问到d对象,因为block修饰的属性是assgin,block保存在栈中,所以对d对象的引用计数(retaincount)不会加1,d对象释放之后,stu对用了block就会访问到僵尸对象,导致程序崩溃。

**解决方法**

typedefvoid(^myBlock)(void);@interfaceStudent:NSObject*注意:如果是block使用copy并不是拷贝,而是转移*@property(nonatomic,copy)myBlock pBlock;@property(nonatomic,copy)NSString*name;@end

将block修饰符变为copy的时候,block会被转移到堆中,会对其引用的对象增加引用计数。

3.copy修饰block的循环引用问题

2019-01-2715:51:57.716059+0800CopyUserProperty[10115:1259494]d=<Dog:0x1005168e0>2019-01-2715:51:57.716220+0800CopyUserProperty[10115:1259494]-[Student dealloc]Program endedwithexit code:0

上面的代码中Student对象可以释放,但dog对象没有释放。

**解除方法**

只需要在block使用的对象前面加上__block修饰

__block Dog*d=[[Dog alloc]init];Student*stu=[[Student alloc]init];stu.pBlock=^{NSLog(@"d = %@",d);};[d release];stu.pBlock();[stu release];

2019-01-2716:04:59.728277+0800CopyUserProperty[10441:1288724]-[Dog dealloc]2019-01-2716:04:59.728615+0800CopyUserProperty[10441:1288724]d=<Dog:0x100682b30>2019-01-2716:04:59.728632+0800CopyUserProperty[10441:1288724]-[Student dealloc]Program endedwithexit code:0

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 前言 Blocks是C语言的扩充功能,而Apple 在OS X Snow Leopard 和 iOS 4中引入了这...
    小人不才阅读 3,786评论 0 23
  • 1.设计模式是什么? 你知道哪些设计模式,并简要叙述?设计模式是一种编码经验,就是用比较成熟的逻辑去处理某一种类型...
    龍飝阅读 2,194评论 0 12
  • block.png iOS代码块Block 概述 代码块Block是苹果在iOS4开始引入的对C语言的扩展,用来实...
    全栈农民工阅读 601评论 0 1
  • Block使用场景,可以在两个界面的传值,也可以对代码封装作为参数的传递等。用过GCD就知道Block的精妙之处。...
    Coder_JMicheal阅读 739评论 2 1
  • 端午节,你一个人吗?不怕,我陪你。如果没法遇见,那也没关系,我的心,为你留在这里。 这是工作后在外过的第二个端午节...
    半夏听海阅读 294评论 0 0