Objective-c之NSCopying

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;}

时间: 2024-11-11 18:26:47

Objective-c之NSCopying的相关文章

iOS开发核心语言Objective C —— 全部知识点总结

本分享是面向有意向从事iOS开发的伙伴及苹果产品的发烧友,亦或是已经从事了iOS的开发人员,想进一步提升者.假设您对iOS开发有极高的兴趣,能够与我一起探讨iOS开发.一起学习,共同进步.假设您是零基础.建议您先翻阅我之前分享的iOS开发分分钟搞定C语言系列,然后在開始Objective C语言的学习,假设您遇到问题也能够与我探讨.另外将无偿分享自己整理出来的大概400G iOS学习视频及学习资料,都是干货哦.能够新浪微博私信?关注极客James,期待与您的共同学习和探讨! ! 由于时间有限,每

Objective C 快速入门学习五

<一>继承和多态 @class Complex 声明类(同C++) 子类函数成员 super 访问父类 同C++类似 1.通过继承 在子类中添加新方法 2.通过继承 在子类中添加新成员 3.通过继承 实现多态(实现比较简单,通过Id通用类型作为父类) 4.重载 5.抽象类abstract作用:创建子类更容易:提供了处理所有派生子类的公共接口:抽象方法制定了标准协议,规范子类必须实现. 6.通用类型id,编译时不会做类型检查,在运行时才会动态绑定具体类型,指出错误. 静态类型在编译阶段就会指出错

iOS开发——新特性OC篇&amp;Objective新特性

Objective新特性 Overview 自 WWDC 2015 推出和开源 Swift 2.0 后,大家对 Swift 的热情又一次高涨起来,在羡慕创业公司的朋友们大谈 Swift 新特性的同时,也有很多像我一样工作上依然需要坚守着 Objective-C 语言的开发者们.今年的 WWDC 中介绍了几个 Objective-C 语言的新特性,还是在“与 Swift 协同工作”这种 Topic 里讲的,越发凸显这门语言的边缘化了,不过有新特性还是极好的,接下来,本文将介绍下面三个主要的新特性:

C++ vs Objective C

oc Short list of some of the major differences: C++ allows multiple inheritance, Objective-C doesn't.一个允许多继承,一个不允许Unlike C++, Objective-C allows method parameters to be named and the method signature includes only the names and types of the parameter

Qt for iOS,Qt 与Objective C混合编程

项目设置 既然要聊 Qt 混合 OC 编程,首先要简单介绍一下 Objective C .我只有一句话:Go,问搜索引擎去.因为我所知实在有限,怕误导了您.当然如果您不怕,往下看吧. OC源文件介绍 首先我要说一下 Objective C 的源文件,后缀是.m 或 .mm ,在 .mm 文件里,可以直接使用 C++ 代码.所以,我们要混合 Qt 代码与 OC 代码,就需要在 Qt 项目里加入 mm 文件. pro 文件配置 Qt SDK for Mac ,安装之后, Qt Creator 会使用

K-means: optimization objective(最小化cost function来求相应的参数)

类似于linear regression,K-means算法也optimization objective或者是试图使cost function求最小值. 了解K-means算法的optimization objective有助于我们(1)调试算法时,看算法是否运行正确(在本节中可以看到)(2)使算法找到更好的cluster,避免局部最优解(在下节中会讲) K-means optimization objective uc(i):表示x(i)分给的那个cluster的cluster centro

objective - c 类和对象

知识点 1.初步认识 OC 2.xcode使用 3.面向对象编程思想,类和对象 4.类的定义和使用 5.类的实例化--对象 6.声明和实现相分离 =========================================== 1.需要理解的(理论部分) 第一部分:OC和面向对象 1.1.OC语言的历史,特点.重要程度 [什么是OC] objective - c   在C语言基础上开发, 面向对象的语言: 注意C语言和oc语言语法的区别: IOS, iphone, ipad … [进入OC

Objective C (iOS) for Qt C++ Developers(iOS开发,Qt开发人员需要了解什么?)

Qt/C++开发人员眼中的Obj-C 对于我们第一次自己定义iOS应用来说,对于来自Qt/C++开发人员来说,我不得不学习Objective-C相关语法与知识 为了让读者可以更easy理解这篇博客的内容,我将描写叙述诸多我所学到的东西.这将很多其它的是大脑转存而不是单一的教程 ,所以我仍然希望这将对你非常有帮助,我将首先写编程语言的差异,然后再写关于类库的差异 1.Objective C vs C vs C++ 类似于C++,Obj-c是C语言的一个超集(这不是100%正确的,可是一个足够好的语

Objective - c Chapter 1 -2 Hello world

Objective - c   Chapter 1  Hello world 1.1 1.2.On the Welcome screen, click "Create a new Xcode project" (see Figure 2-1), or just chooseFile ->New ->New Project. 1.3.在main里写如下代码 #import <Foundation/Foundation.h> int main(int argc, c

Objective C笔记(第一天)

• OC语言概述 1.早在20世纪80年代早期,Bard Cox发明了Objective C, 是扩充的C,面向对象的编程语言. 2.NEXTSTEP简称NS a.1985年,Steve Jobs成?立了NeXT公司. b.1988年,NeXT SoftWare公司获得了OC的语言授权, NEXTSTEP环境成为了苹果操作系统,创建的开发包.MACOS c.1996年,12月20?日,苹果公司宣布收购了NeXT software公司,NEXTSTEP环境为apple公司下主要开发.发?行操作系统