进入本主题之前首先需要了解一些准备知识!
那就是地址与指针的关系:计算机的内存储器被划分为一个个的内存单元。内存单元按照一定的规则编号,这个编号就是存储单元的地址。地址编码的最基本单位是字节(一个字节由8个二进制位组成),也就是说每个字节就是一个基本内存单元,对应一个地址。计算机就是通过这种地址编码的方式来管理内存数据的,进行读写的操作。
在程序中通过两种方式实现对内存的存取:一是通过变量,二是通过内存空间的地址。
具有静态生存期的变量在程序开始运行前,也就是在编译的时候就已经分配了内存空间:具有动态生存期的变量,在程序运行时遇到变量的声明语句才被分配内存空间。在变量获取内存空间的同时,变量名也就成为了对应内存空间的名称,代表内存中的数据,所以可以通过变量名来访问内存空间。
如果是动态分配的内存单元,则根本就没有名称,只能通过地址访问。例如:在不同函数之间传递大量数据时,如果不是传递变量值,而是传递变量地址,就会减少系统开销,提高效率。
动态内存分配:是指在程序执行过程中,动态地分配或者回收存储空间的分配内存的方法。不像数组那样需要预先分配存储空间,而是由系统根据程序的需要分配,且大小就是程序要求的大小。最常用的就是链表进行动态分配内存。
好了开始进入正题;
在对于自定义的对象支持copy功能,也就是我们要给自定义的对象发送copy message,那我们就要手动实现NSCopying协议。在项目开发中我们如果对某个字典或者数组对象进行了一次mutbleCopy其实系统默认的调用了如下API:
- (id)mutableCopyWithZone:(nullable NSZone *)zone;
首先大家看一下这段代码的测试结果(复制的别人的加上自己的见解)
对于copy/mutableCopy NSString
NSString *string = @"hahahhahha";
// string的%p打印输出的是地址(存储内容是@"hahahhahha")
//string的%@打印输出的是存储内容"hahahhahha"
栈区地址string=0x10001040
堆区内容string=@"hahahhahha";
NSString *copyString = [string copy];
// 没有产生新对象(系统没有再开辟一块"hahahhahha"的内存空间,只是在栈上开辟了一个内存空间,用于存储"hahahhahha"的地址)
NSMutableString *mutableCopyString = [string mutableCopy];
// 产生新对象(系统会在堆区再开辟一块"hahahhahha"的内存空间,并且在栈上开辟了一个内存空间,用于存储"新对象的内存空间"的地址)
NSLog(@"string = %p copyString = %p mutableCopyString = %p", string, copyString, mutableCopyString);
string=0x100001040 copyString=0x100001040 mutableCopyString=0x10020e320
对于copy/mutableCopy NSMutableString
NSMutableString *string = [NSMutableString stringWithString:@"嘿嘿"];
// string的%p打印输出的是地址(存储内容是@"嘿嘿")
//string的%@打印输出的是存储内容"嘿嘿"
栈区地址string=0x100206930
堆区内容string=@“嘿嘿”
NSString *copyString = [string copy];
// 产生新对象(系统会在堆区再开辟一块"嘿嘿"的内存空间,并且在栈上开辟了一个内存空间,用于存储"新对象的内存空间"的地址)
NSMutableString *mutableCopyString = [string mutableCopy];
// 产生新对象(系统会在堆区再开辟一块"嘿嘿"的内存空间,并且在栈上开辟了一个内存空间,用于存储"新对象的内存空间"的地址)
NSLog(@"string = %p copyString = %p mutableCopyString = %p", string, copyString, mutableCopyString);
string=0x100206930 copyString=0x100206a80 mutableCopyString=0x100206ab0
对于其它的对象nsarray,nsdictionary,nsmutablearray,nsmutabledictionary同样适用
上面的拷贝类型的理解:浅拷贝(指针拷贝)是指在栈区开辟了一块内存,存储了和原对象地址一样的地址,这两个栈区中的地址指向同一个堆区中的对象。说的通俗一点就是浅拷贝相当于对指针进行了一次retain操作,引用计数器加1,多了一个指针指向共同的对象。
深拷贝(内容拷贝)是指不仅在堆区新开辟了一块内存,存储对象,并且还在栈区开辟了一块内存,存储了和原对象不一样的地址,新开辟栈区存储的地址就是在堆区新开辟对象内存空间的地址。
总结:对于不可变对象的copy,都是浅拷贝,拷贝出来的对象都是不可变对象;对于可变对象的copy,都是深拷贝,拷贝出来的对象是不可变对象;对于不可变对象的mutablecopy,都是深拷贝,拷贝出来的对象都是可变对象;对于可变对象的mutablecopy,都是深拷贝,拷贝出来的对象都是可变对象。想要让自定义的对象支持copy和mutableCopy那么就要实现NSCopying协议,和NSMutableCopying协议。
copy与strong的区别
@interface ZPerson : NSObject
@property (nonatomic, readwrite,copy) NSString *name;
@end
NSMutableString *string = [NSMutableString stringWithFormat:@"11111"];
ZPerson *person = [[ZPerson alloc] init];
person.name = string;
//person.name = string实际是对name这样操作了
//-(void)setName:(NSString*)name MRC下
{
if(_name){
[_name release];
}
_name=[name copy];//此处copy是深拷贝,则person.name的地址是在栈区新开辟内存中存储的地址,和string的地址不一样,因此所指向的对象就不一样,对string的操作不影响person.name的内容。
}
[string appendString:@" hans"];
NSLog(@"name = %@", person.name);
输出内容name=11111
@property (nonatomic, strong) NSString *name;
NSMutableString *string = [NSMutableString stringWithFormat:@"11111"];
ZPerson *person = [[ZPerson alloc] init];
person.name = string;
// ARC下 因为name是strong类型的,person.name = string;相当于对string浅拷贝,引用计数加1,string和person.name指向同一对象,因此对string的操作就相当于对person.name的操作。
此处copy是深拷贝,则person.name的地址是在栈区新开辟内存中存储的地址,和string的地址不一样,因此所指向的对象就不一样,对string的操作不影响person.name的内容。
[string appendString:@" hans"];
NSLog(@"name = %@", person.name);
输出内容name=11111hans