Objective-c之NSCopying

 
 

copy的原理:

执行<NSCopying>协议,类中必须实现copyWithZone:方法响应的copy消息。

copy消息将发送copyWithZone:这个消息给你的类,它的参数是nil。

copyWithZone:返回一个不可改变的copy

  1. *MyPoint.h*/
  2. #import <Foundation/Foundation.h>
  3. @interface MyPoint : NSObject<NSCopying>//继承超类
  4. {
  5. int x;
  6. int y;
  7. };
  8. //set方法
  9. -(void) setMyPoint:(int)x andY:(int)y;
  10. //get方法
  11. -(int) getX;
  12. -(int) getY;
  13. @end
[cpp] view plaincopy
  1. /*MyPoint.m*/
  2. #import "MyPoint.h"
  3. @implementation MyPoint
  4. -(void) setMyPoint:(int)_x andY:(int)_y
  5. {
  6. //要对传进来的x和y进行判断
  7. if(_x<0)
  8. {
  9. x = 1;
  10. }
  11. if(_y<0)
  12. {
  13. y = 1;
  14. }
  15. x = _x;
  16. y = _y;
  17. }
  18. -(int) getX
  19. {
  20. return x;
  21. }
  22. -(int) getY
  23. {
  24. return y;
  25. }
  26. //copy
  27. /*
  28. zone参数处理不同的你alloc出来的内存区域,如果你写的应用程序alloc了大量的内存,并且你又想优化你的内存区域。
  29. 你可以给copywithzone传值,调用allocwithzone来alloc内存:这个方法可以在指定的区域alloc内存
  30. */
  31. -(id) copyWithZone:(NSZone *)zone //创建一个复制的接收器,储存zone
  32. {
  33. MyPoint *newPoint = [[MyPoint allocWithZone:zone]init];
  34. [newPoint setMyPoint:x andY:y];
  35. return newPoint;
  36. }
  37. /*
  38. NSZone 是苹果系统对内存分配和释放的优化方式。
  39. NSZone不是一个对象;它是一个C结构,用于纪录关于内存处理(管理)一系列对象的信息
  40. 在这里它处理了zone这个传进来的对象的信息
  41. */
  42. @end
[cpp] view plaincopy
  1. #import <Foundation/Foundation.h>
  2. #import "MyPoint.h"
  3. int main (int argc, const char * argv[])
  4. {
  5. MyPoint *point1 = [[MyPoint alloc]init];//创建myPoint对象,alloc是分配内存空间,init是初始化
  6. [point1 setMyPoint:2 andY:3];//调用对象point1的方法
  7. NSLog(@"x = %i",[point1 getX]);//[point1 getX]调用get方法
  8. NSLog(@"y = %i",[point1 getY]);
  9. MyPoint *point2 = [point1 copy];//实现复制构造
  10. [point2 setMyPoint:5 andY:5];
  11. NSLog(@"x = %i",[point2 getX]);
  12. NSLog(@"y = %i",[point2 getY]);
  13. [point1 release];// release 是释放分配的内存空间
  14. [point2 release];
  15. return 0;
  16. }
  17. ios拷贝小议

     

    1.copy vs mutableCopy

    copy,对于不可变的对象,简单的指向其内存.对于可变对象,复制内存内容到新的内存中并把新的内存值赋值给左值.

    mutableCopy,始终复制到新的内存中,以一个可变的类型赋值给左值.

    2.copy vc retain

    retain,引用计数+1,内存地址赋值给左值.

    copy,对于不可变对象的,相当于retain;对于可变对象,则是深拷贝赋值.

    举例:

        NSString* a = [NSString stringWithFormat:@"%@",@"this is a"];
    
        NSString* b = [a copy];
    
        NSString* bb = [a retain];
    
        NSString* cc = [a mutableCopy];//实际上cc应该是NSMutableString类型
    
        NSLog(@"%d,%d,%d,%d",[a retainCount],[b retainCount],[bb retainCount],[cc retainCount]);
    
    //输出3,3,3,1

        NSMutableString* a = [NSMutableString stringWithFormat:@"%@",@"this is a"];
    
        NSString* b = [a copy];//不可变的b
    
        NSString* bb = [a retain];//实际类型是NSMutableString的bb
    
        NSString* cc = [a mutableCopy];//同上
    
        NSLog(@"%d,%d,%d,%d",[a retainCount],[b retainCount],[bb retainCount],[cc retainCount]);
    
    //输出2,1,2,1

    3.一些问题

    通过上面2点,思考下面的问题

    我们通常如果这样定义一个变量

    @property(nonatomic,copy)  NSMutableString* mString;

    然后这样使用

    @synthesize mString;
    
    NSMutableString* a = [NSMutableString stringWithFormat:@"%@",@"this is a"];
    
    self.mString = a;
    
    [mString insertString:@"m-" atIndex:0];

    能通过么?当然不能,赋值后的mString是NSString类型的,不可变.如果需要可以改变就需要自己定义属性函数.

    -(void)setMString:(NSMutableString *)m
    
    {
    
        mString = [m mutableCopy];
    
    }
    
    -(NSMutableString *)mString
    
    {
    
        return mString;
    
    }

    (当然,NSMutableString不是线程安全的,一般都建议私有之:@private;或者一定要用的话以NSString作为对外接口类型)

    4.NSCopying NSMutableCopying NSCopyObjective()

    NSCopying就是复制一个对象

    NSMutableCopying就是深拷贝一个对象,让两个对象的改变互不影响

    (其实上面着两个完全看你怎么写啦)

    NSCopyObject(self,0,zone)就是简单的赋值=

    (在涉及到ns对象的时候,NSCopyObject不建议使用)

    注意看下面一个例子:

    @interface ClassB : NSObject <NSCopying>{    NSString* stringB;}
    
    @property(nonatomic,copy) NSString* stringB;
    
    @end

    -(id)copyWithZone:(NSZone *)zone
    
    {
    
        ClassB *b = NSCopyObject(self, 0, zone);
    
    // 使用NSCopyObject时的正确赋值方法,因为没有涉及到原来的内存指针什么事
    
        b->stringB = @"what";
    
    // 看看被注释的这个错误方法,由于setter方法的特性,原来的stringB指向的内存的retainCount减一
    
    // 而由于NSCopyObject的特性,两者又是指向同一个地址的,所以,原类中stirngB指向的地址已经释放了,之后你dealloc中在释放一次?!.就出错啦
    
    //    b.stringB = @"what";
    
        return b;
    
    }
    ios拷贝小议
    2011-10-13 10:44:45     我来说两句       
    收藏    我要投稿

    1.copy vs mutableCopy

    copy,对于不可变的对象,简单的指向其内存.对于可变对象,复制内存内容到新的内存中并把新的内存值赋值给左值.

    mutableCopy,始终复制到新的内存中,以一个可变的类型赋值给左值.

    2.copy vc retain

    retain,引用计数+1,内存地址赋值给左值.

    copy,对于不可变对象的,相当于retain;对于可变对象,则是深拷贝赋值.

    举例:

    NSString* a = [NSString stringWithFormat:@"%@",@"this is a"];    NSString* b = [a copy];    NSString* bb = [a retain];    NSString* cc = [a mutableCopy];//实际上cc应该是NSMutableString类型    NSLog(@"%d,%d,%d,%d",[a retainCount],[b retainCount],[bb retainCount],[cc retainCount]);//输出3,3,3,1    NSMutableString* a = [NSMutableString stringWithFormat:@"%@",@"this is a"];    NSString* b = [a copy];//不可变的b    NSString* bb = [a retain];//实际类型是NSMutableString的bb    NSString* cc = [a mutableCopy];//同上    NSLog(@"%d,%d,%d,%d",[a retainCount],[b retainCount],[bb retainCount],[cc retainCount]);//输出2,1,2,1
    3.一些问题

    通过上面2点,思考下面的问题

    我们通常如果这样定义一个变量

    @property(nonatomic,copy)  NSMutableString* mString;然后这样使用

    @synthesize mString;NSMutableString* a = [NSMutableString stringWithFormat:@"%@",@"this is a"];self.mString = a;[mString insertString:@"m-" atIndex:0];
    能通过么?当然不能,赋值后的mString是NSString类型的,不可变.如果需要可以改变就需要自己定义属性函数.

    -(void)setMString:(NSMutableString *)m{    mString = [m mutableCopy];}-(NSMutableString *)mString{    return mString;}(当然,NSMutableString不是线程安全的,一般都建议私有之:@private;或者一定要用的话以NSString作为对外接口类型)

    4.NSCopying NSMutableCopying NSCopyObjective()

    NSCopying就是复制一个对象

    NSMutableCopying就是深拷贝一个对象,让两个对象的改变互不影响

    (其实上面着两个完全看你怎么写啦)

    NSCopyObject(self,0,zone)就是简单的赋值=

    (在涉及到ns对象的时候,NSCopyObject不建议使用)

    注意看下面一个例子:

    @interface ClassB : NSObject <NSCopying>{    NSString* stringB;}@property(nonatomic,copy) NSString* stringB;@end-(id)copyWithZone:(NSZone *)zone{    ClassB *b = NSCopyObject(self, 0, zone);// 使用NSCopyObject时的正确赋值方法,因为没有涉及到原来的内存指针什么事    b->stringB = @"what";// 看看被注释的这个错误方法,由于setter方法的特性,原来的stringB指向的内存的retainCount减一// 而由于NSCopyObject的特性,两者又是指向同一个地址的,所以,原类中stirngB指向的地址已经释放了,之后你dealloc中在释放一次?!.就出错啦//    b.stringB = @"what";    return b;}

转载于:https://www.cnblogs.com/iOS-mt/p/4119663.html

Objective-c之NSCopying相关推荐

  1. iOS全栈式开发工程师

    课程目录: --/iOS全栈式开发工程师/ ├──1.双师班课程介绍 | └──1.什么是双师教育模式.flv 13.19M ├──10.C语言之字符串 | └──1.字符串.flv 175.41M ...

  2. Objective C范型

    范型 范型编程是一种程序语言设计范式,它允许程序员在使用强类型的语言编写代码的时候,延迟确定具体的类型. 以Swift代码为例,假如有一个需求是要交换两个int,很容易写出类似代码 func swap ...

  3. Objective C内存管理之理解autorelease------面试题

    Objective C内存管理之理解autorelease Autorelease实际上只是把对release的调用延迟了,对于每一个Autorelease,系统只是把该Object放入了当前的Aut ...

  4. Objective C浅拷贝和深拷贝

    ##浅拷贝 浅拷贝就是对内存地址的复制,让目标对象指针和源对象指向同一片内存空间.如: char* str = (char*)malloc(100); char* str2 = str; 复制代码 浅 ...

  5. iOS完全自学手册——[三]Objective-C语言速成,利用Objective-C创建自己的对象...

    1.前言 上一篇已经介绍了App Delegate.View Controller的基本概念,除此之外,分别利用storyboard和纯代码创建了第一个Xcode的工程,并对不同方式搭建项目进行了比较 ...

  6. Objective C 中的nil,Nil,NULL和NSNull理解

    kenyo网友的原创说法是:做IOS开发的估计都对Objective-C的内存管理机制很头疼,一不小心程序就会出内存泄露,我也不例外,前几天被指针的置nil与release给搞惨了,今和大家详细解说一 ...

  7. 【iOS 开发】Objective-C 运算符

    博客地址 : http://blog.csdn.net/shulianghan/article/details/41624613 参考文章 : 1.[iOS 开发]Object-C 运算符 2.[iO ...

  8. 第22条:理解NSCopying 协议

    本条要点:(作者总结) 若想令自己所写的对象具有拷贝功能,则需实现 NSCopying 协议. 如果自定义的对象分为可变版本与不可变版本,那么就要同时实现 NSCopying 与 NSMutableC ...

  9. 成功解决coordinate_descent.py:491: ConvergenceWarning: Objective did not converge. You might want to inc

    成功解决coordinate_descent.py:491: ConvergenceWarning: Objective did not converge. You might want to inc ...

最新文章

  1. 常见问题:内存,循环引用,runloop的简单理解
  2. Socket技术详解(一篇就够了)
  3. jwt php tp5,TP5框架中使用JWT的方法示例
  4. “管理压力,控制情绪”培训小结
  5. linux是一个工程而不是理论
  6. Java ArrayList到数组
  7. ffmpeg命令_温故知新:ffmpeg操作《天空之城》。窗口党勿入,都是指令!
  8. Windows 关于Robocopy的使用详解
  9. 485通讯的校验和_案例丨MCGS与变频器、温度控制器进行通讯演示
  10. php国际象棋棋盘奇行奇列,国际象棋怎么玩
  11. 计算机毕业设计指导 教师 评语,毕业设计指导教师评语【集锦】
  12. ThreeJs DRACO压缩并加载gltf模型
  13. JavaScript多线程编程
  14. Juniper Junos设置3层接口
  15. Google Pixel 刷机、OpenGApps和系统分区扩容
  16. Java:File和IO流
  17. 一个小Demo带你理解安卓事件分发
  18. WordPress部分函数详解
  19. Oracle 后台进程(六)PMON进程
  20. illustrator下载_使用Illustrator和Photoshop创建复仇者联盟文字效果

热门文章

  1. 跨库查询(OpenDataSource)与链接服务器(Linking Server)
  2. 转载:LINQ to SQL更新数据库操作
  3. DataObjects.NET -- A Excellent O/R Mapping Framework!
  4. 4E4 models
  5. WHEN OVERSEAS
  6. marting mate 是真的香喷喷的 学术英语必备
  7. 好的英文视频照片网站
  8. 积极的活下去本身就是挺好的一件事情了
  9. release,debug库互调用,32位,64位程序与库互调用
  10. oracle使用存储过程做铺底数据