iOS中野指針問題敘述

野指針就是指向一個已刪除的對象或者受限內(nèi)存區(qū)域的指針讳窟。我們寫C++的時(shí)候強(qiáng)調(diào)指針初始化為NULL柳骄,強(qiáng)調(diào)用完后也為其賦值為NULL病梢,誰分配的誰回收瘦赫,來避免野指針的問題辰晕。比較常見的就是這個指針指向的內(nèi)存,在別處被回收了确虱,但是這個指針不知道含友,依然還指向這塊內(nèi)存。MRC 時(shí)代因?yàn)橐糜?jì)數(shù)手動控制校辩,所以內(nèi)存很容易在別處被回收窘问。ARC解決了大部分這種問題。宜咒、在iOS9之前惠赫,系統(tǒng)庫的delegate和target-action有一部分是assign(unsafe_unretain)的形式,這時(shí)候如果內(nèi)存在別處被回收了故黑,也是會出現(xiàn)野指針的儿咱。所以iOS9之后這些地方就改成了weak內(nèi)存修飾符庭砍,內(nèi)存被回收的時(shí)候通過weak表,把這些指針設(shè)為nil混埠。也大幅度減少了野指針的出現(xiàn)怠缸。

如果現(xiàn)在在工程中依然頻繁出現(xiàn)野指針,幾乎可以肯定是錯誤地使用了內(nèi)存钳宪。

比較常見的就是這個指針指向的內(nèi)存揭北,在別處被回收了,但是這個指針不知道使套,依然還指向這塊內(nèi)存

野指針指向的內(nèi)存沒有被覆蓋的時(shí)候罐呼,或者被覆蓋成可以訪問的內(nèi)存的時(shí)候,不一定會出現(xiàn)崩潰侦高。這個時(shí)候向?qū)ο蟀l(fā)送消息嫉柴,不一定會崩潰(可能剛好有這個方法),或者向已經(jīng)釋放的對象發(fā)送消息奉呛。 但是如果野指針指向的是僵尸對象计螺,那就一定會崩潰了,會崩潰在僵尸對象第一次被其它消息訪問的時(shí)候瞧壮。

iOS9之前的delegate 崩潰

在iOS9之前的tableview的delegate和datasource都是assign內(nèi)存修飾符的登馒。iOS9之后才使用weak。

// iOS 8 之前@property(nonatomic,assign)id dataSource@property(nonatomic,assign)id delegate// iOS 9 之后@property(nonatomic,weak,nullable)id dataSource@property(nonatomic,weak,nullable)id delegate

這種情況咆槽,如果delegate比tableview本身更早被釋放陈轿,此時(shí)的dataSource就會成為一個野指針。常見的情況比如block調(diào)用延長了tableview的生命周期秦忿,就可能會發(fā)生這種情況麦射,導(dǎo)致野指針crash。 一般崩潰日志里是objc_msgsend + 15的崩潰,崩潰在delegate或者datasource的方法里灯谣。

解決方法也很簡單潜秋,在dealloc的時(shí)候把dataSource和delegate設(shè)為nil即可。

- (void)dealloc

{

? ? _tableView.delegate = nil;

? ? _tableView.dataSource = nil;

}

野指針定位有幾個關(guān)鍵:

第一是意識到這是野指針的問題:Mach Exception大多數(shù)都是野指針的問題胎许,崩潰日志里最多見objc_msgSend和unrecognized selector sent to等等峻呛。而且往往跟iOS SDK版本和iphone型號有關(guān)。 認(rèn)識到野指針的問題后辜窑,就不必要拘泥于崩潰日志钩述,因?yàn)楸罎⒌牡胤诫x崩潰的原因比較遠(yuǎn)了。

第二是盡可能重現(xiàn)穆碎。利用Zombie Object/Scribble/Aasn都可以切距。個人認(rèn)為自己實(shí)現(xiàn)的Zombie Object最好,既可以脫離Xcode debug的限制惨远,使用又比較簡單谜悟。

第三是根據(jù)野指針指向的對象來判斷出錯的位置,而不是崩潰的方法北秽。因?yàn)楸罎⒌姆椒x崩潰的原因比較遠(yuǎn)了葡幸,但是野指針指向的對象多半還是出錯的對象(有時(shí)也可能被覆蓋了)。

第四是利用malloc stack/lzMalloc找到野指針指向?qū)ο蟪跏蓟奈恢煤蚫ealloc的位置贺氓,判斷是否過早釋放等蔚叨。

空指針 野指針 僵尸對象

空指針:

1. 沒有存儲任何內(nèi)存地址的指針就稱為空指針(NULL指針)。

2.被賦值為nil的指針辙培,在沒有被具體初始化之前蔑水,為nil。

注意:?

nil和Null區(qū)別不是初始化前后的區(qū)別扬蕊,是nil代表對象類型的空指針搀别,Null代表基本數(shù)據(jù)類型的空指針。

3.nil尾抑、Nil歇父、NULL、NSNULL的含義和區(qū)別

nil:OC中的對象的空指針

Nil:OC中類的空指針

NULL:C類型的空指針

NSNull:數(shù)值類的空對象

此處說一下NSNull再愈,在集合中不能nil值榜苫,因?yàn)镹SArray和NSDictionary中nil有特殊的含義。但是有些時(shí)候翎冲,需要在集合中存放空值垂睬,比如個人信息中,只知道姓名抗悍,不知道電話號碼驹饺,此時(shí),有必要將電話號碼設(shè)置為空檐春,這時(shí)逻淌,就用到了NSNull。

NSNull中只有一個null方法 :[NSNull null]

可以給空指針發(fā)送消息疟暖,不會造成crash

野指針:

1."野指針"不是nil指針卡儒,是指向"垃圾"內(nèi)存(不可用內(nèi)存)的指針。野指針是非常危險(xiǎn)的俐巴。

示例:

Student *stu = [[Student alloc] init];

[stu setAge:10];

[stu release];這里已經(jīng)釋放內(nèi)存

[stu setAge:10];---》報(bào)錯

如果改動一下代碼骨望,就不會報(bào)錯

Student *stu = [[Student alloc] init];

[stu setAge:10];

[stu release];

stu = nil;?

[stu setAge:10]; //消息是無法發(fā)送出去的,不會造成任何的影響欣舵,當(dāng)然也不會報(bào)錯擎鸠。

補(bǔ)充說明:

1.Student對象接收到release消息后,會馬上被銷毀缘圈,所占用的內(nèi)存會被回收劣光⊥嗖希” 這里執(zhí)行release只是標(biāo)記對象占用的那塊內(nèi)存可以被釋放,但是具體的釋放的時(shí)間是不可控的绢涡,如果在release之后執(zhí)行[stu setAge:10];不一定會野指針crash牲剃,如果對象內(nèi)存已經(jīng)被其他對象覆寫占用,那么會crash雄可,如果沒有沒覆寫凿傅,調(diào)用依然可以正確執(zhí)行。

2.向空指針發(fā)送消息不會報(bào)錯数苫,但是給野指針發(fā)送消息會報(bào)錯

僵尸對象

遇到exc_bad_access這類問題一般都是僵尸對象引起的聪舒,可以開啟僵尸模式定位,我們并沒有保留他虐急,只是在程序運(yùn)行到該對象的時(shí)候會產(chǎn)生問題箱残,沒有誰會運(yùn)用他,只會定位他然后解決掉

內(nèi)存回收的本質(zhì).

1.申請一塊空間,實(shí)際上是向系統(tǒng)申請一塊別人不再使用的空間.

2.釋放一塊空間,指的是占用的空間不再使用,這個時(shí)候系統(tǒng)可以分配給別人去使用.

3.在這個個空間分配給別人之前 數(shù)據(jù)還是存在的.

? ? 3.1.OC對象釋放以后,表示OC對象占用的空間可以分配給別人.

? ? 3.2.但是再分配給別人之前 這個空間仍然存在 對象的數(shù)據(jù)仍然存在.

4.僵尸對象: 一個已經(jīng)被釋放的對象 就叫做僵尸對象.

使用野指針訪問僵尸對象.有的時(shí)候會出問題,有的時(shí)候不會出問題.

1.當(dāng)野指針指向的僵尸對象所占用的空間還沒有分配給別人的時(shí)候,這個時(shí)候其實(shí)是可以訪問的.

因?yàn)閷ο蟮臄?shù)據(jù)還在.

2.當(dāng)野指針指向的對象所占用的空間分配給了別人的時(shí)候 這個時(shí)候訪問就會出問題.

3.所以,你不要通過一個野指針去訪問一個僵尸對象.

? ? ? ?3.1.雖然可以通過野指針去訪問已經(jīng)被釋放的對象,但是我們不允許這么做.

僵尸對象檢測.

1.默認(rèn)情況下. Xcode不會去檢測指針指向的對象是否為一個僵尸對象. 能訪問就訪問 不能訪問就報(bào)錯.

2.可以開啟Xcode的僵尸對象檢測.

? ? ? ?2.1.那么就會在通過指針訪問對象的時(shí)候,檢測這個對象是否為一個僵尸對象 如果是僵尸對象 就會報(bào)錯.

為什么不默認(rèn)開啟僵尸對象檢測呢?

1.因?yàn)橐坏╅_啟,每次通過指針訪問對象的時(shí)候.都會去檢查指針指向的對象是否為僵尸對象.那么這樣的話 就影響效率了.

如何避免僵尸對象報(bào)錯.

1.當(dāng)一個指針變?yōu)橐爸羔樢院? 就把這個指針的值設(shè)置為nil

僵尸對象無法復(fù)活.

1.當(dāng)一個對象的引用計(jì)數(shù)器變?yōu)?以后 這個對象就被釋放了.

2.就無法取操作這個僵尸對象了. 所有對這個對象的操作都是無效的.

3.因?yàn)橐坏ο蟊换厥?對象就是1個僵尸對象 而訪問1個僵尸對象 是沒有意義.

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末戏仓,一起剝皮案震驚了整個濱河市疚宇,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌赏殃,老刑警劉巖敷待,帶你破解...
    沈念sama閱讀 206,126評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異仁热,居然都是意外死亡榜揖,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評論 2 382
  • 文/潘曉璐 我一進(jìn)店門抗蠢,熙熙樓的掌柜王于貴愁眉苦臉地迎上來举哟,“玉大人,你說我怎么就攤上這事迅矛》列桑” “怎么了?”我有些...
    開封第一講書人閱讀 152,445評論 0 341
  • 文/不壞的土叔 我叫張陵秽褒,是天一觀的道長壶硅。 經(jīng)常有香客問我,道長销斟,這世上最難降的妖魔是什么庐椒? 我笑而不...
    開封第一講書人閱讀 55,185評論 1 278
  • 正文 為了忘掉前任,我火速辦了婚禮蚂踊,結(jié)果婚禮上约谈,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好棱诱,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,178評論 5 371
  • 文/花漫 我一把揭開白布泼橘。 她就那樣靜靜地躺著,像睡著了一般军俊。 火紅的嫁衣襯著肌膚如雪侥加。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 48,970評論 1 284
  • 那天粪躬,我揣著相機(jī)與錄音,去河邊找鬼昔穴。 笑死镰官,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的吗货。 我是一名探鬼主播泳唠,決...
    沈念sama閱讀 38,276評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼宙搬!你這毒婦竟也來了笨腥?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 36,927評論 0 259
  • 序言:老撾萬榮一對情侶失蹤勇垛,失蹤者是張志新(化名)和其女友劉穎脖母,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體闲孤,經(jīng)...
    沈念sama閱讀 43,400評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡谆级,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,883評論 2 323
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了讼积。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片肥照。...
    茶點(diǎn)故事閱讀 37,997評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖勤众,靈堂內(nèi)的尸體忽然破棺而出舆绎,到底是詐尸還是另有隱情,我是刑警寧澤们颜,帶...
    沈念sama閱讀 33,646評論 4 322
  • 正文 年R本政府宣布吕朵,位于F島的核電站,受9級特大地震影響掌桩,放射性物質(zhì)發(fā)生泄漏边锁。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,213評論 3 307
  • 文/蒙蒙 一波岛、第九天 我趴在偏房一處隱蔽的房頂上張望茅坛。 院中可真熱鬧,春花似錦、人聲如沸贡蓖。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,204評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽斥铺。三九已至彻桃,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間晾蜘,已是汗流浹背邻眷。 一陣腳步聲響...
    開封第一講書人閱讀 31,423評論 1 260
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留剔交,地道東北人肆饶。 一個月前我還...
    沈念sama閱讀 45,423評論 2 352
  • 正文 我出身青樓,卻偏偏與公主長得像岖常,于是被迫代替她去往敵國和親驯镊。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,722評論 2 345

推薦閱讀更多精彩內(nèi)容

  • Swift1> Swift和OC的區(qū)別1.1> Swift沒有地址/指針的概念1.2> 泛型1.3> 類型嚴(yán)謹(jǐn) 對...
    cosWriter閱讀 11,089評論 1 32
  • 1.設(shè)計(jì)模式是什么竭鞍? 你知道哪些設(shè)計(jì)模式板惑,并簡要敘述? 設(shè)計(jì)模式是一種編碼經(jīng)驗(yàn)偎快,就是用比較成熟的邏輯去處理某一種類...
    司馬DE晴空閱讀 1,277評論 0 7
  • 1.設(shè)計(jì)模式是什么滨砍? 你知道哪些設(shè)計(jì)模式往湿,并簡要敘述?設(shè)計(jì)模式是一種編碼經(jīng)驗(yàn)惋戏,就是用比較成熟的邏輯去處理某一種類型...
    龍飝閱讀 2,138評論 0 12
  • 成因 野指針就是指向一個已刪除的對象或者受限內(nèi)存區(qū)域的指針领追。我們寫C++的時(shí)候強(qiáng)調(diào)指針初始化為NULL,強(qiáng)調(diào)用完后...
    上官soyo閱讀 20,835評論 4 80
  • 【人民網(wǎng)評《王者榮耀》:是娛樂大眾還是"陷害"人生】作為游戲《王者榮耀》是成功的,而面向社會舔亭,它卻不斷在釋放負(fù)能量...
    律兜閱讀 550評論 0 0