转自:http://blog.csdn.net/java886o/article/details/9046273

  • #import <Foundation/Foundation.h>
  • int main(int argc, const char * argv[])
  • {
  • @autoreleasepool {
  • NSMutableArray* arr1 = [[NSMutableArray alloc] initWithObjects:@"1",@"2",@"3",@"4",@"5", nil];
  • //使用retain得到对象所有权,arr1的引用计数加1后为2
  • NSMutableArray* arr2 = [arr1 retain];
  • NSLog(@"arr1当前引用计数:%ld",arr1.retainCount);
  • NSLog(@"arr2当前引用计数:%ld",arr2.retainCount);
  • NSLog(@"arr1数组元素:%@",arr1);
  • NSLog(@"arr2数组元素:%@",arr2);
  • NSLog(@"删除arr2最后一个元素,观察arr1和arr2的元素");
  • [arr2 removeLastObject];
  • NSLog(@"arr1数组元素:%@",arr1);
  • NSLog(@"arr2数组元素:%@",arr2);
  • //使用Copy返回不可变的对象
  • NSMutableArray* arr3 = [arr1 copy];
  • //[arr3 removeLastObject];  错误写法,因为Copy返回的是不可变对象,因此不能对对象进行修改删除元素操作
  • NSLog(@"arr1数组元素:%@\t%p",arr1,arr1);
  • NSLog(@"arr3数组元素:%@\t%p",arr3,arr3);
  • //使用MutaleCopy返回可变的对象
  • NSMutableArray* arr4 = [arr1 mutableCopy];
  • [arr4 removeLastObject]; //可变的对象可以进行删除修改操作
  • NSLog(@"arr1数组元素:%@\t%p",arr1,arr1);
  • NSLog(@"arr4数组元素:%@\t%p",arr4,arr4);
  • NSArray* abc = [[NSArray alloc]initWithObjects:@"111",@"222", nil];
  • NSArray* def = [abc copy];
  • NSLog(@"abc 的地址:%p",abc);
  • NSLog(@"def 的地址:%p",def);
  • /*
  • 复制对象总结:
  • 1.Foundation框架中的可复制对象,通过上面的代码``得到结论: 当对不可变的对象调用Copy时,实际上并未产生新的对象,还是指向原来的对象的内存空间,作用相当于retain
  • 2.只要满足:被拷贝的对象是可变的或调用的方法是MutableCopy方法,那么产生的对象就是一个新的对象,这是就和retain不一样了
  • 2.retain 方法返回的只是原对象的引用,并未产生一个新的对象,新的对象引用指向了原对象引用指向的同一个内存空间.
  • 3.Copy 方法返回的对象是一个不可变对象
  • 4.MutableCopy 方法返回的对象是一个可变的对象
  • 5.无论原对象是可变的还是不可变的,使用Copy方法得到的新对象都是不可变的
  • 6.无论原对象是可变的还是不可变的,使用MutableCopy方法得到的新对象都是可变的
  • */
  • //测试Foundation框架中支持拷贝的对象是浅拷贝还是深拷贝
  • NSMutableArray* test = [[NSMutableArray alloc]init];
  • for (int i = 0;i < 3; i++) {
  • NSObject* obj = [[NSObject alloc]init];
  • [test addObject:obj];
  • [obj release];
  • }
  • NSLog(@"拷贝之前:test\n\n");
  • for (NSObject* obj in test) {
  • NSLog(@"%@\t%p\t%ld",obj,obj,obj.retainCount);
  • }
  • NSMutableArray* result = [test copy];
  • NSLog(@"拷贝之后:test\n\n");
  • for (NSObject* obj in test) {
  • NSLog(@"%@\t%p\t%ld",obj,obj,obj.retainCount);
  • }
  • NSLog(@"拷贝之后:result\n\n");
  • for (NSObject* obj in result) {
  • NSLog(@"%@\t%p\t%ld",obj,obj,obj.retainCount);
  • }
  • /*
  • 深浅复制对象总结:
  • 1.Foundation框架中复制的类,默认是浅复制.
  • 2.浅复制只复制对象本身,对象里所包含的属性,包含的对象不复制,换言之原对象和新对象的包含的属性或对象都指向同一个内存空间.
  • 2.深复制既复制对象本身,对象里所包含的属性,包含的对象也复制,换言之原对象和新对象所包含的属性或对象完全不相同,指向了2个不同的内存空间.
  • */
  • /*
  • 自定义对象的复制总结:
  • 1.对象要拥有拷贝特性,需要实现NSCopying或NSMutableCopying协议,并实现协议所对应的copyWithZone:方法或mutableCopyWithZone:方法
  • 2.NSCopying和NSMutableCopying可只实现其中一个,或2个都实现
  • 3.NSCopying 实现返回的对象是不可变的
  • 4.NSMutableCopying 实现返回的对象是可变的
  • 5.浅复制标准实现写法如下:
  • - (id) copyWithZone:(NSZone*) zone {
  • Person* p = [[[self class] allocWithZone:zone] init];
  • p.name = name;
  • p.age = age;
  • return p;
  • }
  • - (id) mutableCopyWithZone:(NSZone*) zone {
  • Person* p = [[[self class] allocWithZone:zone] init];
  • p.name = name;
  • p.age = age;
  • return p;
  • }
  • 6.深复制标准写法如下:
  • - (id) copyWithZone:(NSZone*) zone {
  • Person* p = [[[self class] allocWithZone:zone] init];
  • p.name = [name copy];
  • p.age = [age copy];
  • return p;
  • }
  • - (id) mutableCopyWithZone:(NSZone*) zone {
  • Person* p = [[[self class] allocWithZone:zone] init];
  • p.name = [name MutableCopy];
  • p.age = [age copy];
  • return p;
  • }
  • */
  • }
  • return 0;
  • }

【IPHONE开发-OBJECTC入门学习】复制对象,深浅复制的更多相关文章

  1. 【IPHONE开发-OBJECTC入门学习】对象的归档和解归档

    转自:http://blog.csdn.net/java886o/article/details/9046967 #import <Foundation/Foundation.h> #im ...

  2. 【IPHONE开发-OBJECTC入门学习】文件的操作,读写复制文件

    转自:http://blog.csdn.net/java886o/article/details/9041547 FileTools.h FileTools.m #import "FileT ...

  3. java复制对象,复制对象属性,只可复制两个对象想同的属性名。也可自定义只复制需要的属性。

    注意:使用时copy()方法只会复制相同的属性.常用的copy()方法.以下为封装的工具和使用方式. 1.封装类 import java.util.Map; import java.util.Weak ...

  4. HoloLens开发手记 - 入门学习阶段总结

    伴随着数月的期待,终于拿到了预订的HoloLens开发者版本套件.随着VR/AR/MR技术的热潮,国内外均对它们的应用与盈利前景持有积极的预期,这也直接导致了国内外当前投资VR/AR/MR技术的热潮. ...

  5. 树莓派开发板入门学习笔记1:[转]资料收集及树莓派系统在Ubuntu安装

    参考教程(微雪课堂):http://www.waveshare.net/study/portal.php 树莓派实验室: http://shumeipai.nxez.com/2014/12/21/us ...

  6. 迅为iTOP-4412物联网开发板入门学习高手进阶项目开发超树莓派

    免费视频教程: 为初学者精心录制的整套视频教程全部免费,随IT技术发展而不断增添的视频教程仍然免费!一支有经验的工程师团队会始终成为您的后盾. 项目实战---全开源: 手机远程控制开发板 门禁系统 W ...

  7. eclipse开发android入门学习

    1.device窗体            查看android执行情况的窗体和DDMS内容一致 2.android.bat             在adb路径下执行adb操作,实现android系统 ...

  8. LARK BOARD开发板入门学习-第2篇

    1. 本次主要研究下HDMI接口,使用芯片是CH7033,这个芯片可以接VGA和HDMI两种接口,和FPGA的接口是地址数据总线 2. 值得注意的地方,下图的D1,双二极管BAT54S在电路中一般用于 ...

  9. 树莓派开发板入门学习笔记2:[转]树莓派系统在VM中能做什么

    问"树莓派系统在VM中能做什么"不如问"树莓派能做什么":(参考:树莓派实验室) 普通难度的DIY 较高难度的DIY 用树莓派打造一个家庭影院 给树莓派安装摄像 ...

随机推荐

  1. java读取文本文件内容

    版权声明:本文为xing_star原创文章,转载请注明出处! 本文同步自http://javaexception.com/archives/128 java读取文本文件内容 今天写代码写着要调试一个很 ...

  2. Socket简单实现ssh笔记

    Scoket概念: socket本质上就是在2台网络互通的电脑之间,架设一个通道,两台电脑通过这个通道来实现数据的互相传递. 我们知道网络 通信 都 是基于 ip+port 方能定位到目标的具体机器上 ...

  3. Java并发编程艺术读书笔记

    1.多线程在CPU切换过程中,由于需要保存线程之前状态和加载新线程状态,成为上下文切换,上下文切换会造成消耗系统内存.所以,可合理控制线程数量. 如何控制: (1)使用ps -ef|grep appn ...

  4. [20190531]ORA-600 kokasgi1故障模拟与恢复.txt

    [20190531]ORA-600 kokasgi1故障模拟与恢复.txt --//昨天看链接:http://www.xifenfei.com/2019/05/ora-600-kokasgi1-rec ...

  5. 7.2 Spark Streaming

    一.Spark Streaming设计 Spark Streaming可整合多种输入数据源,如Kafka.Flume.HDFS,甚至是普通的TCP套接字.经处理后的数据可存储至文件系统.数据库,或显示 ...

  6. 17.Java基础_初探类的private和public关键字

    package pack1; public class Student { // 成员变量 private String name; private int age; // get/set方法 pub ...

  7. 7. Go语言—时间模块

    一.时间模块 1. 统计程序执行时间 package main import ( "time" "fmt" ) func test() { time.Sleep ...

  8. 7、zabbix自定义监控阈值-前端页面报警

    找个值监控一下: #监控passwd #默认是间隔是1小时,我们改成10秒,下面我们要把报警打开 #我们在被监控上的主机上创建一个新用户,过10秒,界面上就会报警了 ----------------- ...

  9. vlmcsd

    scp ./vlmcsd-x64-musl-static xxx@host.ip:/opt/kms/ chmod u+x /opt/kms/vlmcsd-x64-musl-static ./vlmcs ...

  10. LG2495 「SDOI2011」消耗战 虚树

    问题描述 LG2495 题解 虚树 \(\mathrm{Code}\) #include<bits/stdc++.h> using namespace std; #define int l ...