淺拷貝:拷貝對象本身,返回一個對象指針竭沫,指向相同的內(nèi)存地址。
深拷貝:拷貝內(nèi)容本身,返回一個對象指針骑篙,指向不同的內(nèi)存地址蜕提。
obj2 = [obj1 copy]返回的一個不可變對象,無論obj1是可變對象還是不可變對象靶端。如果obj1是一個不可變對象谎势,那么它們指向同一個對象。obj2 = [obj1 mutableCopy]返回的是一個可變對象杨名,無論obj1是可變對象還是不可變對象脏榆。即使obj1也是一個可變對象,它們?nèi)灾赶虿煌刂诽ǖ莾蓚€對象姐霍。
源對象類型拷貝方法對象類型是否產(chǎn)生新對象拷貝類型
NSMutableStringcopyNSStringYES深拷貝(內(nèi)容拷貝)
mutableCopyNSMutableStringYES深拷貝(內(nèi)容拷貝)
NSStringcopyNSStringNO淺拷貝(指針拷貝)
mutableCopyNSMutableStringYES深拷貝(內(nèi)容拷貝)
NSArraycopyNSArrayNO淺拷貝(指針拷貝)
mutableCopyYES深拷貝(內(nèi)容拷貝)
注意:其他對象NSArray、NSMutableArray 、NSDictionary镊折、NSMutableDictionary一樣適用
HSPerson *p = [[HSPerson alloc] init];
p.age = 20;
p.height = 170.0;
HSPerson *copyP = [p copy]; # 這里崩潰
看崩潰信息HSPerson應(yīng)該先實現(xiàn):
- (id)copyWithZone:(NSZone *)zone;//NSCopying協(xié)議方法
測試:
#import "HSPerson.h"
@interface HSPerson()<NSCopying>
@end
@implementation HSPerson
- (id)copyWithZone:(NSZone *)zone
{
? ? return @"測試copyWithZone";
}
@end
HSPerson *p = [[HSPerson alloc] init];
p.age = 20;
p.height = 170.0;
HSPerson *copyP = [p copy];
NSLog(@"copyP: %@", copyP);
可以看出copyWithZone重新分配新的內(nèi)存空間,則:
- (id)copyWithZone:(NSZone *)zone
{
? ? HSPerson *person = [[HSPerson allocWithZone:zone] init];
? ? return person;
// 有些人可能下面alloc,重新初始化空間介衔,但這方法已給你分配了zone恨胚,自己就無需再次alloc內(nèi)存空間了
//? ? HSPerson *person = [[HSPerson alloc] init];
}
HSPerson *p = [[HSPerson alloc] init];
p.age = 20;
p.height = 170.0;
HSPerson *copyP = [p copy];
NSLog(@"p = %p copyP = %p", p, copyP);
NSLog(@"age = %d height = %f", copyP.age, copyP.height);
雖然copy了份新的對象,然而age炎咖,height值并未copy赃泡,那么:
- (id)copyWithZone:(NSZone *)zone
{
? ? HSPerson *person = [[HSPerson allocWithZone:zone] init];
? ? person.age = self.age;
? ? person.height = self.height;
? ? // 這里self其實就要被copy的那個對象,很顯然要自己賦值給新對象乘盼,所以這里可以控制copy的屬性
? ? return person;
}
- (id)mutableCopyWithZone:(NSZone *)zone
{
? ? HSPerson *person = [[HSPerson allocWithZone:zone] init];
? ? person.age = self.age;
? ? person.height = self.height;
? ? return person;
}
property里的copy升熊、strong區(qū)別
copy
#import <Foundation/Foundation.h>
@interface HSPerson : NSObject
@property (nonatomic, copy) NSString *name;
@end
NSMutableString *string = [NSMutableString stringWithFormat:@“測試"];
HSPerson *person = [[HSPerson alloc] init];
person.name = string;
// 不能改變person.name的值,因為其內(nèi)部copy新的對象
[string appendString:@" hans"];
?NSLog(@"name = %@", person.name);
property copy 實際上就對name干了這個:
- (void)setName:(NSString *)name
{
? ? _name = [name copy];
}
@property (nonatomic, copy) NSMutableString *name;
對應(yīng):
- (void)setName:(NSMutableString *)name
{
? ? _name = [name copy];
}
copy出來的仍然是不可變字符绸栅!如果有人用NSMutableString的方法级野,就會崩潰:
strong
@property (nonatomic, strong) NSString *name;
NSMutableString *string = [NSMutableString stringWithFormat:@"測試"];
HSPerson *person = [[HSPerson alloc] init];
person.name = string;
// 可以改變person.name的值,因為其內(nèi)部沒有生成新的對象
[string appendString:@" hans"];
NSLog(@"name = %@", person.name);
總結(jié):用copy與strong取決于需求粹胯,如果不希望被外界更改用copy蓖柔,反之用strong