IOS 屬性修飾符 assign weak copy

assign

弱引用
適用于基本數(shù)據(jù)類型 int float double bool

weak

弱引用
適用于 NSObject 對象,weak修飾的對象在釋放之后,指針地址會被置為nil。
runtime 對注冊的類會進(jìn)行布局,對于weak對象會放入一個hash表中。用weak指向的對象內(nèi)存地址作為key,當(dāng)此對象的引用計(jì)數(shù)器為0時會dealloc,加入weak指向的對象內(nèi)存地址是a,那么就會以a為鍵,在這個weak表中搜索,找到所有以a為鍵的weak對象,并設(shè)置為nil

copy

為什么 NSString NSArray NSDictionary 經(jīng)常使用copy?
使用copy的目的是為了讓對象的屬性不受外界的影響。假如傳入的是NSMutableArray,使用copy后就變成了NSArray,外界不能改變。

NSError 使用copy因?yàn)槠鋵?shí)現(xiàn)了copying協(xié)議

這樣寫有什么問題: @property (copy) NSMutableArray *array;

  1. 添加,刪除數(shù)組內(nèi)的元素會因?yàn)檎也坏綄?yīng)的方法而崩潰,因?yàn)閏opy就是復(fù)制了一個不可變NSArray對象。
  2. 使用atomic 嚴(yán)重影響性能

深拷貝 淺拷貝
對系統(tǒng)非容器類不可變對象調(diào)用copy方法其實(shí)只是把當(dāng)前對象的指針指向了原對象的地址,而調(diào)用mutableCopy方法則是新分配了一塊內(nèi)存區(qū)域并把新對象的指針指向了這塊區(qū)域。
對于可變對象來說,調(diào)用copy和mutableCopy方法都會重新分配一塊內(nèi)存,但對可變對象調(diào)用copy方法會返回不可變對象。
但重新改變淺復(fù)制對象的值,會變成深拷貝

NSString *str = @"123";
NSString *strCopy = [str copy]; // 淺拷貝
    
NSLog(@"str = %@    strCopy = %@",str,strCopy);
NSLog(@"str地址:%p    strCopy地址:%p",str,strCopy);
str = @"a"; // 改變值會真正的實(shí)現(xiàn)拷貝(深拷貝)
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

對于不可變?nèi)萜鲗ο笳{(diào)用copy方法只是增加了對原對象的指針的引用,調(diào)用mutableCopy方法是重新分配一塊內(nèi)存,然后把新對象指針指向新內(nèi)存。
而對于可變對象不管是調(diào)用copy還是mutableCopy都是新分配一塊內(nèi)存。但雖然重新分配了一塊內(nèi)存,但是對象里面的數(shù)據(jù)依然是指針賦值。

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
)

可以看到當(dāng)更改原數(shù)組的值之后,所有新數(shù)組的值都更改了,即使調(diào)用了MutableCopy方法創(chuàng)建的新數(shù)組里面的值也因此更改,所以可以看出對于系統(tǒng)容器類對象,其元素對象始終是指針復(fù)制。

自己定義的類默認(rèn)沒有實(shí)現(xiàn)NSCopying協(xié)議

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(實(shí)例變量) + getter + setter

@implementation Book
@dynamic  name;
@synthesize age = _age;

@synthesize
編譯器自動生成 getter/setter 方法
當(dāng)有自定義的存取方法時,不會自動生成。

@dynamic
告訴編譯器不自動生成 getter/setter 方法

在protocol中使用@property只會生成setter getter方法,目的是希望遵守協(xié)議的對象能實(shí)現(xiàn)該屬性方法
當(dāng)同時重寫了setter和getter時,系統(tǒng)就不會生成ivar。解決方法有下面兩種

@interface ViewController () {
    // NSString *_name; 方法1:手動創(chuàng)建ivar
}
@property (nonatomic, strong) NSString *name;
@end

@implementation ViewController
//@synthesize name = _name; 方法2:使用@sysnthesize關(guān)聯(lián)@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"; // 調(diào)用 setter
NSLog(@"%@",self.name); // 調(diào)用 getter

_name = @"cat"; // 什么都不調(diào)用
NSLog(@"%@",_name); // 什么都不調(diào)用
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容