assign
弱引用
適用于基本數據類型 int float double bool
weak
弱引用
適用于 NSObject 對象,weak修飾的對象在釋放之后,指針地址會被置為nil。
runtime 對注冊的類會進行布局,對于weak對象會放入一個hash表中。用weak指向的對象內存地址作為key,當此對象的引用計數器為0時會dealloc,加入weak指向的對象內存地址是a,那么就會以a為鍵,在這個weak表中搜索,找到所有以a為鍵的weak對象,并設置為nil
copy
為什么 NSString NSArray NSDictionary 經常使用copy?
使用copy的目的是為了讓對象的屬性不受外界的影響。假如傳入的是NSMutableArray,使用copy后就變成了NSArray,外界不能改變。
NSError 使用copy因為其實現了copying協議
這樣寫有什么問題: @property (copy) NSMutableArray *array;
- 添加,刪除數組內的元素會因為找不到對應的方法而崩潰,因為copy就是復制了一個不可變NSArray對象。
- 使用atomic 嚴重影響性能
深拷貝 淺拷貝
對系統非容器類不可變對象調用copy方法其實只是把當前對象的指針指向了原對象的地址,而調用mutableCopy方法則是新分配了一塊內存區域并把新對象的指針指向了這塊區域。
對于可變對象來說,調用copy和mutableCopy方法都會重新分配一塊內存,但對可變對象調用copy方法會返回不可變對象。
但重新改變淺復制對象的值,會變成深拷貝
NSString *str = @"123";
NSString *strCopy = [str copy]; // 淺拷貝
NSLog(@"str = %@ strCopy = %@",str,strCopy);
NSLog(@"str地址:%p strCopy地址:%p",str,strCopy);
str = @"a"; // 改變值會真正的實現拷貝(深拷貝)
NSLog(@"str = %@ strCopy = %@",str,strCopy);
NSLog(@"str地址:%p strCopy地址:%p",str,strCopy);
NSMutableString *mustr = [str copy]; // mustr 已是不可變對象
NSMutableString *muCopy = [str mutableCopy];
NSLog(@"mustr = %@ muCopy = %@",mustr,muCopy);
NSLog(@"mustr地址:%p muCopy地址:%p",mustr,muCopy);
[muCopy appendString:@"b"];
NSLog(@"mustr = %@ muCopy = %@",mustr,muCopy);
NSLog(@"mustr地址:%p muCopy地址:%p",mustr,muCopy);
str = 123 strCopy = 123
str地址:0x106ecdc58 strCopy地址:0x106ecdc58
str = a strCopy = 123
str地址:0x106ecdcb8 strCopy地址:0x106ecdc58
mustr = a muCopy = a
mustr地址:0x106ecdcb8 muCopy地址:0x60000007cc80
mustr = a muCopy = ab
mustr地址:0x106ecdcb8 muCopy地址:0x60000007cc80
對于不可變容器對象調用copy方法只是增加了對原對象的指針的引用,調用mutableCopy方法是重新分配一塊內存,然后把新對象指針指向新內存。
而對于可變對象不管是調用copy還是mutableCopy都是新分配一塊內存。但雖然重新分配了一塊內存,但是對象里面的數據依然是指針賦值。
NSArray *array = [NSArray arrayWithObjects:[NSMutableString stringWithString:@"1"],@"2",@"3", nil];
NSArray *arrayCopy = [array copy];
NSArray *arrayMutableCopy = [array mutableCopy];
NSLog(@"array地址:%p arrayCopy地址:%p arrayMutableCopy地址:%p",array,arrayCopy,arrayMutableCopy);
NSLog(@"array = %@ arrayCopy = %@ arrayMutableCopy = %@",array,arrayCopy,arrayMutableCopy);
NSMutableArray *mutableArrayCopy = [array copy];
NSMutableArray *mutableArrayMutableCopy = [array mutableCopy];
NSLog(@"array地址:%p mutableArrayCopy地址:%p mutableArrayMutableCopy地址:%p",array,mutableArrayCopy,mutableArrayMutableCopy);
NSLog(@"array = %@ mutableArrayCopy = %@ mutableArrayCopy = %@",array,mutableArrayCopy,mutableArrayMutableCopy);
NSMutableString *mustr = array[0];
[mustr appendString:@"2"];
NSLog(@"array地址%p arrayCopy地址%p arrayMutableCopy地址%p mutableArrayCopy地址%p mutableArrayCopy地址%p",array,arrayCopy,arrayMutableCopy,mutableArrayCopy,mutableArrayMutableCopy);
NSLog(@"array = %@ arrayCopy = %@ arrayMutableCopy = %@ mutableArrayCopy = %@ mutableArrayCopy = %@",array,arrayCopy,arrayMutableCopy,mutableArrayCopy,mutableArrayMutableCopy);
array地址:0x6080000538f0 arrayCopy地址:0x6080000538f0 arrayMutableCopy地址:0x608000050d70
array = (
1,
2,
3
)
arrayCopy = (
1,
2,
3
)
arrayMutableCopy = (
1,
2,
3
)
array地址:0x6080000538f0 mutableArrayCopy地址:0x6080000538f0 mutableArrayMutableCopy地址:0x608000050890
array = (
1,
2,
3
)
mutableArrayCopy = (
1,
2,
3
)
mutableArrayCopy = (
1,
2,
3
)
array地址0x6080000538f0 arrayCopy地址0x6080000538f0 arrayMutableCopy地址0x608000050d70 mutableArrayCopy地址0x6080000538f0 mutableArrayCopy地址0x608000050890
array = (
12,
2,
3
)
arrayCopy = (
12,
2,
3
)
arrayMutableCopy = (
12,
2,
3
)
mutableArrayCopy = (
12,
2,
3
)
mutableArrayCopy = (
12,
2,
3
)
可以看到當更改原數組的值之后,所有新數組的值都更改了,即使調用了MutableCopy方法創建的新數組里面的值也因此更改,所以可以看出對于系統容器類對象,其元素對象始終是指針復制。
自己定義的類默認沒有實現NSCopying協議
Person *p1 = [[Person alloc] init];
Person *p2 = [p1 copy]; //崩潰
*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[Person copyWithZone:]: unrecognized selector sent to instance 0x608000011570'
property
@property = ivar(實例變量) + getter + setter
@implementation Book
@dynamic name;
@synthesize age = _age;
@synthesize
編譯器自動生成 getter/setter 方法
當有自定義的存取方法時,不會自動生成。
@dynamic
告訴編譯器不自動生成 getter/setter 方法
在protocol中使用@property只會生成setter getter方法,目的是希望遵守協議的對象能實現該屬性方法
當同時重寫了setter和getter時,系統就不會生成ivar。解決方法有下面兩種
@interface ViewController () {
// NSString *_name; 方法1:手動創建ivar
}
@property (nonatomic, strong) NSString *name;
@end
@implementation ViewController
//@synthesize name = _name; 方法2:使用@sysnthesize關聯@property 與ivar
- (void)viewDidLoad {
[super viewDidLoad];
}
- (NSString *)name{
//return _name;
}
- (void)setName:(NSString *)name{
//_name = @"";
}
@end
set get
- (void)setName:(NSString *)name{
if (_name != name) {
[name retain];
[_name release];
_name = name;
}
}
self.name = @"dog"; // 調用 setter
NSLog(@"%@",self.name); // 調用 getter
_name = @"cat"; // 什么都不調用
NSLog(@"%@",_name); // 什么都不調用