概述


  • 拷贝:复制一个与源对象内容相同的对象
  • 实现拷贝,需要遵守以下两个协议

    • NSCopying
    • NSMutableCopying
  • 拷贝返回对象的种类

    • 可变,mutableCopy消息返回的对象
    • 不可变,copy消息返回的对象
  • 拷贝的种类

    • 浅拷贝,只是复制了一个指向源对象的指针,未创建对象,未分配内存
    • 深拷贝,复制了源对象,创建了新对象,分配了内存
  • 注意

    • 系统对容器类的对象与非容器类的对象的内存处理是不同的,即当一个没有被其他对象强引用的对象从容器中移除后,该对象就销毁

Copy与Retain


  • copy

    • 是创建一个新的对象,内容拷贝
    • copy表示的是两个对象的内容相同, 新对象的引用计数为1
    • 与旧对象的引用计数无关,就对象没有变化
    • copy减少了对象对上下文的
  • retain

    • 创建的是一个指针,指针拷贝
    • 对象地址相同,内容固然相同
    • 对象的引用计数+1

不同对象的拷贝行为


  • 非容器对象(如NSString))

    • 对于不可变对象

      • 规则

        • copy,浅拷贝(指针复制)
        • mutableCopy,深拷贝(对象复制),返回对象可变(产生新的 可变对象)
      • 示例

        - (void)imutableInstanceCopy
        {
        NSString *string = @"Welcome to Xcode"; //copy,浅拷贝
        NSString *stringCopy = [string copy];
        //mutableCopy,返回的对象可变
        NSMutableString *stringMutableCopy = [string mutableCopy];
        [stringMutableCopy appendString:@"!"]; //string与stringCopy的内存地址相同
        NSLog(@"string: %p", string);
        NSLog(@"strongCopy: %p", stringCopy); //string与stringMutableCopy的内存地址不同,分配了新的内存
        NSLog(@"stringMCopy:%p", stringMutableCopy);
        }
    • 对于可变对象

      • 规则

        • copy,深拷贝(对象复制),返回对象不可变
        • mutableCopy,深拷贝(对象复制)
      • 示例

        - (void)mutableInstanceCopy
        {
        NSMutableString *mutableString = [NSMutableString stringWithString: @"Welcome to Xcode"]; //深拷贝,返回对象不可变
        NSString *stringCopy = [mutableString copy];
        NSMutableString *mutableStringCopy = [mutableString copy];
        //运行时,此句会报错,错误信息“Attempt to mutate immutable object with appendString:”
        [mutableStringCopy appendString:@"~~~"]; //深拷贝,返回对象可变
        NSMutableString *stringMutableCopy = [mutableString mutableCopy];
        [stringMutableCopy appendString:@"!"]; //三者与mutableString的内存地址都不同
        NSLog(@"mutableString: %p", mutableString);
        NSLog(@"string: %p", stringCopy);
        NSLog(@"mutableStringCopy: %p", mutableStringCopy);
        NSLog(@"stringMutbleCopy:%p", stringMutableCopy);
        }
  • 容器对象(NSArray

    • 遵循非容器对象的拷贝原则
    • 注意

      • 容器内的元素是指针赋值(浅拷贝)
      • 示例

        - (void)containerInstanceShallowCopy
        {
        NSArray *array = [NSArray arrayWithObjects:[NSMutableString stringWithString:@"Welcome"],@"to",@"Xcode",nil]; //浅拷贝
        NSArray *arrayCopy = [array copy];
        //深拷贝
        NSMutableArray *arrayMutableCopy = [array mutableCopy]; NSLog(@"array: %p", array);
        NSLog(@"arrayCopy: %p", arrayCopy);
        NSLog(@"arrayMutableCopy: %p", arrayMutableCopy); //容器内的对象是浅拷贝,即它们在内存中只有一份
        NSMutableString *testString = [array objectAtIndex:0];
        [testString appendString:@" you"]; //三个数组的内容同时改变
        NSLog(@"array[0]: %@", array[0]);
        NSLog(@"arrayCopy[0]: %@", arrayCopy[0]);
        NSLog(@"arrayMutableCopy[0]: %@", arrayMutableCopy[0]);
        }
    • 实现真正意义上的深复制

      - (void)containerInstanceDeepCopy
      {
      NSArray *array = [NSArray arrayWithObjects:[NSMutableString stringWithString:@"Welcome"],@"to",@"Xcode",nil]; //数组内对象是指针复制
      NSArray *deepCopyArray = [[NSArray alloc] initWithArray:array];
      //真正以上的深复制,数组内对象是对象复制
      NSArray *trueDeepCopyArray = [NSKeyedUnarchiver unarchiveObjectWithData:[NSKeyedArchiver archivedDataWithRootObject:array]]; NSLog(@"array: %p", array);
      NSLog(@"deepCopyArray: %p", deepCopyArray);
      NSLog(@"trueDeepCopyArray: %p", trueDeepCopyArray); //改变array的第一个元素
      [[array objectAtIndex:0] appendString:@" you"]; //只影响deepCopyArray数组的第一个元素
      NSLog(@"array[0]: %@", array[0]);
      NSLog(@"arrayCopy[0]: %@", deepCopyArray[0]);
      //不影响trueDeepCopyArray数组的第一个元素,是真正意义上的深拷贝
      NSLog(@"arrayMutableCopy[0]: %@", trueDeepCopyArray[0]);
      }
  • 自定义对象

    • 在定义对象要实现拷贝,需要遵守NSCoping与NSMutableCoping协议,并实现以下方法

      • - (id)copyWithZone:(NSZone *)zone,可变拷贝
      • - (id)mutableCopyWithZone:(NSZone *)zone,不可变拷贝
    • 示例(自定对象Person的拷贝)

      • 遵守协议,设置成员属性

        @interface Person : NSObject <NSCopying, NSMutableCopying>
        /**姓名*/
        @property (nonatomic, copy) NSMutableString *name;
        /**地址*/
        @property (nonatomic, copy) NSString *address;
        /**年龄*/
        @property (nonatomic, assign) NSInteger age;
        @end
      • 重写初始化方法

        - (instancetype)init
        {
        if (self = [super init])
        {
        self.name = [[NSMutableString alloc] initWithString:@"XiaoYaowang"];
        self.address = @"空山新雨后";
        self.age = 3;
        }
        return self;
        }
      • 实现- (id)copyWithZone:(NSZone *)zone

        - (id)copyWithZone:(NSZone *)zone
        {
        Person *p = [[[self class] allocWithZone:zone] init];
        p.name = [self.name copy];
        p.address = [self.address copy];
        p.age = self.age; return p;
        }
      • 实现- (id)mutableCopyWithZone:(NSZone *)zone

        - (id)mutableCopyWithZone:(NSZone *)zone
        {
        Person *p = [[[self class] allocWithZone:zone] init];
        //注意,此处是mutableCopy方法
        p.name = [self.name mutableCopy];
        p.address = [self.address copy];
        p.age = self.age; return p;
        }

---恢复内容结束---

OC细节 - 1.深拷贝与浅拷贝详解的更多相关文章

  1. 【转】 c++拷贝构造函数(深拷贝,浅拷贝)详解

     c++拷贝构造函数(深拷贝,浅拷贝)详解 2013-11-05 20:30:29 分类: C/C++ 原文地址:http://blog.chinaunix.net/uid-28977986-id-3 ...

  2. 【C++】拷贝构造函数(深拷贝,浅拷贝)详解

    一.什么是拷贝构造函数  首先对于普通类型的对象来说,它们之间的复制是很简单的,例如: ; int b = a; 而类对象与普通对象不同,类对象内部结构一般较为复杂,存在各种成员变量. 下面看一个类对 ...

  3. **Python中的深拷贝和浅拷贝详解

    Python中的深拷贝和浅拷贝详解   这篇文章主要介绍了Python中的深拷贝和浅拷贝详解,本文讲解了变量-对象-引用.可变对象-不可变对象.拷贝等内容.   要说清楚Python中的深浅拷贝,需要 ...

  4. javascript实现引用数据类型的深拷贝和浅拷贝详解

    关于引用类型值的详解,请看另一篇随笔 https://www.cnblogs.com/jinbang/p/10346584.html 深拷贝和浅拷贝,也就是引用数据类型栈和堆的知识点.深浅拷贝的原型都 ...

  5. c++拷贝构造函数(深拷贝,浅拷贝)详解

    一.什么是拷贝构造函数      首先对于普通类型的对象来说,它们之间的复制是很简单的,例如: ; int b=a;   而类对象与普通对象不同,类对象内部结构一般较为复杂,存在各种成员变量.  下面 ...

  6. OC语言-08-深拷贝与浅拷贝详解(示例)

    概述 拷贝:复制一个与源对象内容相同的对象 实现拷贝,需要遵守以下两个协议 NSCopying NSMutableCopying 拷贝返回对象的种类 可变,mutableCopy消息返回的对象 不可变 ...

  7. python的拷贝方式以及深拷贝,浅拷贝详解

    python的拷贝方法有:切片方法, 工厂方法, 深拷贝方法, 浅拷贝方法等. 几种方法都可以实现拷贝操作, 具体区别在于两点:1.代码写法不同. 2.内存地址引用不同 代码演示: import co ...

  8. C++ 深拷贝和浅拷贝详解

    前言 在对象拷贝过程中,如果没有自定义拷贝构造函数,系统会提供一个缺省的拷贝构造函数,缺省的拷贝构造函数对于基本类型的成员变量,按字节复制,对于类类型成员变量,调用其相应类型的拷贝构造函数. 位拷贝( ...

  9. OC中的深拷贝与浅拷贝

    深拷贝(deep copy)与浅拷贝(shallow copy)的定义一直是有争论的. 一种理解是: 所谓的浅拷贝, 就是不完全的拷贝 NSString *s = @"123"; ...

随机推荐

  1. js 获取 sktime时间

    效果图如下: HTML代码: <html> <head> <script> //------------------------------------------ ...

  2. MVC和Web API 过滤器Filter [转]

    ASP.NET MVC 支持以下类型的操作筛选器: ·        授权筛选器.这些筛选器用于实现IAuthorizationFilter和做出关于是否执行操作方法(如执行身份验证或验证请求的属性) ...

  3. 将批量下载的博客导入到手机后,通过豆约翰博客阅读器APP(Android手机)进行浏览,白字黑底,保护眼睛,图文并茂。

    首先下面演示的博文来自于以下地址:http://www.douban.com/note/423939291/ 需要先通过博客备份专家将导出的博文导入到手机(还不会用的朋友请先阅读http://www. ...

  4. KTV点歌系统播放原理

    北大青鸟KTV点歌系统播放原理 指导老师--原玉明 1.首先知道有两个类(PlayList   Song) PlayList类中有一方法 public static bool AddSong(Song ...

  5. pow(x,n) leecode

    https://oj.leetcode.com/problems/powx-n/ 提交地址 快速幂的使用,可以研究一下 public class Solution { public double po ...

  6. DAL与数据库类型的对应关系

    MSSQL Server类型 DbType类型 C#类型   MySql类型 DbType类型 C#类型 bit Boolean bool   bit(1) Boolean bool tinyint ...

  7. gcc “-I”(大写i),“-L”(大写l),“-l”(小写l)的区别

    我们用gcc编译程序时,可能会用到“-I”(大写i),“-L”(大写l),“-l”(小写l)等参数,下面做个记录: 例: gcc -o hello hello.c -I /home/hello/inc ...

  8. git diff old mode 100755 new mode 100644

    755 vs 644 在linux下载了Qt的软件仓库,拷贝了一份到windows下.在 msysgit 下,发现所有的文件都被修改了. 用 git diff 查看,发现是: $ git diff u ...

  9. IT项目管理的六种错误思维

    导读:在软件行业,在界面设计没有正式展现给客户之前,所有的工作都处于需求调研阶段.很多IT项目经理因为年轻,初生牛犊不怕虎,胆量大,勇气足,敢于在实践中引入新的工具.方法.敢于尝试不是坏事,但试验的风 ...

  10. linux内核数据结构--进程相关

    linux里面,有一个结构体task_struct,也叫“进程描述符”的数据结构,它包含了与进程相关的所有信息,它非常复杂,每一个字段都可能与一个功能相关,所以大部分细节不在我的研究范围之内,在这篇文 ...