预备知识 :
内存的栈区 : 由编译器自动分配释放, 存放函数的参数值, 局部变量的值等. 其操作方式类似于数据结构中的栈.
内存的堆区 : 一般由程序员分配释放, 若程序员不释放, 程序结束时可能由OS回收. 注意它与数据结构中的堆是两回事, 分配方式倒是类似于链表.
个人总结:栈区是编译器自动分配与释放内存、堆区是程序员进行分配与释放
copy与mutableCopy
- 不可变的对象采用copy,只是引用源对象的指针,内容未拷贝,没有生成一个新对象;
- 不可变的对象采用mutableCopy,是把指针和内容都进行了拷贝,生成一个新的可变对象;
- 可变的对象采用copy,是把指针和内容都进行了拷贝,生成一个新的不可变对象;
- 可变的对象采用mutableCopy,是把指针和内容都进行了拷贝,生成一个新的可变对象。
# 不可变对象
NSString *string = @"汉斯哈哈哈";
// 没有产生新对象
NSString *copyString = [string copy];
// 产生新对象
NSMutableString *mutableCopyString = [string mutableCopy];
NSLog(@"string = %p copyString = %p mutableCopyString = %p", string, copyString, mutableCopyString);
# 可变对象
NSMutableString *string = [NSMutableString stringWithString:@"汉斯哈哈哈"];
// 产生新对象
NSString *copyString = [string copy];
// 产生新对象
NSMutableString *mutableCopyString = [string mutableCopy];
NSLog(@"string = %p copyString = %p mutableCopyString = %p", string, copyString, mutableCopyString);
对象如何实现深拷贝
// MAPerson.m
- (id)copyWithZone:(NSZone *)zone {
//已给你分配了zone,自己就无需再次alloc内存空间了
MAPerson *person = [[MAPerson allocWithZone:zone] init];
person.name = self.name;
person.age = self.age;
return person;
}
// NSArray类别
- (id)copy
{
NSArray *cpyArray = [[NSArray alloc] initWithArray:self copyItems:YES];
return cpyArray;
}
ps : 官方文档说明copy方法内部默认会调用copyWithZone方法的, 但是NSArray因为未知的原因导致其copy方法不会调用copyWithZone (可能是因为OC中已经废弃了zone这个概念, 苹果官方文档有说), 所以这里我就利用分类重写了NSArray的copy方法, 实际上苹果并不推荐这么做.
属性B赋值给不可变的属性A,属性A采用的是copy的内存修饰符,改变属性B是会影响属性A的值。
属性B赋值给可变的属性A,属性A采用的是copy的内存修饰符,改变属性B是不会影响属性A的值。
集合拷贝
当对集合进行拷贝时,其元素或集合本身的可变性会受到影响。
- copyWithZone: 使表层成为不可变,深层仍保持原状;
- initWithArray:copyItems:NO —— 表层、深层均保持原状;
- initWithArray:copyItems:YES —— 表层保持原状、第二层成为不可变、深层保持原状;
- 压缩\解压 —— 均保持原状;
ps:对数组里的对象进行拷贝,就要遵守 NSCoding 协议并对数组压缩(NSKeyedArchiver)和解压(NSKeyedUnarchiver)。
# 完全拷贝,数组里的对象要实现如下方法
- (void)encodeWithCoder:(NSCoder *)aCoder {
[aCoder encodeObject:self.name forKey:@"name"];
[aCoder encodeInteger:self.age forKey:@"age"];
}
- (instancetype)initWithCoder:(NSCoder *)aDecoder {
self.name = [aDecoder decodeObjectForKey:@"name"];
self.age = [aDecoder decodeIntegerForKey:@"age"];
return self;
}
copy 与 strong
- 对于不可变的对象,不管strong还是copy属性的对象,都是指向源对象,copy只做浅层次的copy。
- 对于可变的对象,strong属性只是增加了源对象的引用计数;copy属性是做了一次深复制,创建了一个新对象,copy属性对象指向新的对象;ps:copy可变的对象,创建是一个不可变的对象。(NSString与NSMutableString)