一.寫在前面的話
為什么要把他排在第一位蛀柴?這是有原因的,因?yàn)檫@破玩意在開發(fā)中很少會(huì)引起注意矫夯,一般人都是一帶而過鸽疾,不去深究那些細(xì)節(jié).
在這里我要給一直在開發(fā)路上奔波的新手們說一句公道話
這里東西,即使不去深究一般的開發(fā)也不會(huì)出現(xiàn)問題训貌,因?yàn)橐粋€(gè)小小的屬性修飾符就否定一個(gè)人制肮,我覺得這已經(jīng)不是草率的問題了冒窍,而是你內(nèi)心的狹隘.
好了接下來我們進(jìn)入正題,屬性修飾符嘛豺鼻,顧名思義就是修飾屬性的综液,而屬性是什么呢?很簡單儒飒,我就用我自己的理解谬莹,就是個(gè)全局變量,在程序每個(gè)地方都可以使用到桩了,那么與成員變量的區(qū)別是什么呢附帽?除了充當(dāng)一個(gè)成員變量以外,就是自動(dòng)生成了一個(gè)setter方法和一個(gè)getter方法井誉,使程序員們方便使用了士葫。
二.屬性修飾符
屬性修飾符都有哪些呢:
strong/retain
assign/weak
copy
atomic/noatomic
readwrite/readonly
strong/retain:
strong是ARC下新引入的屬性修飾符但它在MRC下仍然可以使用,你可以把它當(dāng)成MRC中的retain送悔,用它修飾就是強(qiáng)引對象慢显,retainCount會(huì)加1
假設(shè)有一個(gè)屬性
@property (nonatomic, strong) NSString *name;
在ARC下,這時(shí)它的setter方法應(yīng)該這樣
- (void)setName:(NSString *)name {
_name = name;
}
而MRC下的retain的setter方法是這樣
- (void)setName:(NSString *)name {
//如果地址不一樣,證明_name與name是不同的值
if (_name != name) {
//這時(shí)我們先把舊值釋放 防止內(nèi)存泄露
[_name release];
//然后把新值強(qiáng)引用(retainCount+1)賦值于成員變量上
_name = [name retain];
}
}
assign/weak:
assign:
首先講一下assign
這個(gè)東西是專門用來修飾基本數(shù)據(jù)類型,在iOS開發(fā)中什么是基本數(shù)據(jù)類型呢欠啤?通俗的講就是非對象類型荚藻,比如int、float洁段、double应狱、NSInteger,這種數(shù)字類型就稱為基本數(shù)據(jù)類型祠丝,他們都不是對象疾呻,但是萬事沒有那么絕對的,它也可以修飾對象写半,但是那么做是十分不安全的岸蜗,那么為什么這個(gè)修飾符不能修飾對象呢?那是因?yàn)閍ssign是弱引用叠蝇,如果用它修飾對象璃岳,那么對象一旦創(chuàng)建就會(huì)立即被釋放,然而assign依然會(huì)保留對象的指針悔捶,但是對象已經(jīng)消失了铃慷,所以該指針指向了一塊被釋放的內(nèi)存區(qū)域,即為野指針蜕该,當(dāng)野指針調(diào)用對象方法的時(shí)候(向?qū)ο蟀l(fā)送消息)犁柜,就會(huì)發(fā)生崩潰。
weak:
weak與assign一樣是弱引用堂淡,但是weak是用來修飾對象的馋缅,蘋果規(guī)定不能用它修飾基本數(shù)據(jù)類型坛怪,如果修飾立即就會(huì)報(bào)錯(cuò)
上面說了weak也是弱引用,為什么蘋果會(huì)允許它修飾對象呢股囊,因?yàn)閣eak上面添加了一個(gè)安全機(jī)制袜匿,如果weak所引用的對象被釋放了,就會(huì)把weak上的指針設(shè)置為nil稚疹,這樣當(dāng)對象再執(zhí)行方法的時(shí)候居灯,就不會(huì)發(fā)生崩潰了,因?yàn)榧词箤ο蟊会尫帕四诠罚騨il發(fā)送消息也不會(huì)造成崩潰怪嫌,所以使用weak修飾對象是安全的。
有些人可能不理解為什么要使用weak在這里我給你們拋一塊磚
設(shè)想有這樣一種情況柳沙,當(dāng)一個(gè)子視圖由于某種原因要將它的父控制器作為自己的屬性岩灭,以便于在子視圖的各個(gè)角落都可以調(diào)用調(diào)用到父控制器的方法,這時(shí)如果你使用strong來修飾這個(gè)父控制器就會(huì)出現(xiàn)子視圖強(qiáng)引用父控制器導(dǎo)致父控制器無法被釋放的問題赂鲤,這就是所謂的內(nèi)存泄漏噪径,所以這里需要使用weak來修飾它的父控制器,由于weak是弱引用数初,不會(huì)讓父控制器引用計(jì)數(shù)增加找爱,這樣就既不會(huì)造成內(nèi)存泄漏,又可以優(yōu)雅的使用父控制器對象了泡孩。
copy:
這個(gè)東西往淺了說就是復(fù)制, 往深了說需要說好幾大篇章, 其實(shí)就是在對象進(jìn)行setter方法時(shí)先對對象做復(fù)制操作, 而對可變變量和不可變變量的復(fù)制是不一樣的, 這里總結(jié)一下, 對不可變變量的復(fù)制都是拷貝指針, 也就是常說的淺拷貝, 而對可變變量的copy會(huì)復(fù)制出一個(gè)一模一樣的克隆人放在另一塊內(nèi)存地址中, 只不過該變量從此不可變, 說到這里可能有人對mutableCopy有疑問, 這個(gè)東西不是屬性修飾符, 所以我就順便告訴你一下, 就一句話
只有不可變對象進(jìn)行copy才是淺拷貝, 其他都是深拷貝
下面寫一下copy的setter方法
ARC下
- (void)setName:(NSString *)name {
_name = [name copy];
}
MRC下
- (void)setName:(NSString *)name {
//如果地址不一樣,證明_name與name是不同的值
if (_name != name) {
//這時(shí)我們先把舊值釋放
[_name release];
//然后把新值強(qiáng)引用(retainCount+1)賦值于成員變量
_name = [name copy];
}
}
atomic/noatomic
這兩個(gè)東西我自己也不是特別清楚, 畢竟還是很年輕, 但是你記住atomic是原子性, 使用這個(gè)修飾符可以保證線程安全但是會(huì)降低效率, 而noatomic是非原子性, 不會(huì)保證線程安全, 但效率比atomic高, 我們開發(fā)中一般都是用noatomic.
readwrite/readonly:
先挑簡單的說readwrite就是可讀寫车摄,不加這個(gè)屬性修飾符默認(rèn)就是如此,所有變量都是可以讀寫的仑鸥,也就是都自帶setter和getter方法吮播,而使用readonly修飾系統(tǒng)會(huì)自動(dòng)生成getter方法而不生成setter方法所以是只讀的,無法賦值眼俊。