iOS中深拷贝与浅拷贝

iOS中深拷贝与浅拷贝

浅拷贝

浅拷贝并不是拷贝对象本身,而是对指向对象的指针进行拷贝,但还是指向同一块堆内存中指针指向的对象。

iOS中深拷贝与浅拷贝_第1张图片

  • 可以看出浅拷贝中栈内存中指针对象的地址改变了,但还是指向相同的一块堆内存地址

深拷贝

直接拷贝对象到内存中的一块区域,然后把新对象的指针指向这块内存。

iOS中深拷贝与浅拷贝_第2张图片

  • 可以看出深拷贝是直接拷贝对象到内存中的一块区域(分配了一块新的内存空间),然后把新对象的指针指向这块内存,原对象和被赋值对象互不影响。

深拷贝与浅拷贝的本质区别

在于是否在堆内存中开辟新的内存空间。

注意

所有继承于NSObject的class都声明了copy和mutableCopy,但是并没有实现NSCopying和NSMutableCopying协议,所以Foundation框架中要使用copy方法要实现NSCopying、NSMutableCopying协议的。

区分深拷贝与浅拷贝

主要从 容器类非容器 类着手,不管是容器类还是非容器类都有可变对象和不可变对象,所以分为四种:

  • 非容器类不可变对象:NSString
  • 非容器类可变对象:NSMutableString
  • 容器类不可变对象:NSArray
  • 容器类可变对象:NSMutableArray

Show me the code.

1. 非容器类不可变对象:NSString

iOS中深拷贝与浅拷贝_第3张图片
输出:

可以看出非容器类不可变对象调用copy方法只是把当前对象的指针指向原对象的地址,而调用mutableCopy方法则是分配了一块新的内存区域并把新对象的指针指向这块内存区域。

2. 非容器类可变对象:NSMutableString
iOS中深拷贝与浅拷贝_第4张图片
输出:
在这里插入图片描述

可以看出非容器类可变对象不管是调用copy方法,还是调用mutableCopy方法,都重新开辟了一块新的内存区域。需要注意的是:可变对象调用copy方法后返回的是不可变对象,因此调用方法改变对象的时候程序会崩溃。

3. 容器类不可变对象:NSArray
iOS中深拷贝与浅拷贝_第5张图片
输出:
在这里插入图片描述

可以看出容器类不可变对象调用copy和mutableCopy方法和非容器类不可变对象一样,调用copy方法还是对原对象指针的引用,调用mutableCopy开辟了一块新的内存区域。

4. 容器类可变对象:NSMutableArray
iOS中深拷贝与浅拷贝_第6张图片
输出:
iOS中深拷贝与浅拷贝_第7张图片

可以看出容器类可变对象还是和非容器类可变对象调用copy、mutableCopy方法一样,都开辟了一块新的内存区域。但容器内的元素任然是浅拷贝,因为修改数据后指针指向的对象地址和原对象的地址是一致的。

总结:

[object copy]:对于可变对象为深拷贝,不可变对象为浅拷贝。
[object mutableCopy]:始终都是深拷贝。

你可能感兴趣的:(iOS中深拷贝与浅拷贝)