EXC_BAD_ACCESS的本質詳解以及僵屍模式調試原理

来源:https://www.cnblogs.com/jukaiit/archive/2019/10/28/11754520.html
-Advertisement-
Play Games

原文:What Is EXC_BAD_ACCESS and How to Debug It 有時候,你會遇到由EXC_BAD_ACCESS造成的崩潰。 這篇文章會告訴你什麼是EXC_BAD_ACCESS,以及它產生的原因。我還會提供一些EXC_BAD_ACCESS錯誤的解決方案。 1. 什麼是 EX ...


52f57b4ab65a9t1280l75.jpg

原文:What Is EXC_BAD_ACCESS and How to Debug It

有時候,你會遇到由EXC_BAD_ACCESS造成的崩潰。 這篇文章會告訴你什麼是EXC_BAD_ACCESS,以及它產生的原因。我還會提供一些EXC_BAD_ACCESS錯誤的解決方案。

1. 什麼是 EXC_BAD_ACCESS?

一旦你理解EXC_BAD_ACCESS的本質,你就會更好地理解這個模糊的名詞。這裡有一個極為簡單的解釋,也有一個技術層面的解釋。我們首先從簡單的解釋開始說起。

2. 簡單的解釋

不管什麼時候當你遇到EXC_BAD_ACCESS這個錯誤,那就意味著你向一個已經釋放的對象發送消息。這是最常見的情況,但也有例外,我們將在稍後討論。

3. EXC_BAD_ACCESS的本質

技術層面的解釋有些複雜。在C和Objective-C中,你一直在處理指針。指針無非是存儲另一個變數的記憶體地址的變數。當您向一個對象發送消息時,指向該對象的指針將會被引用。這意味著,你獲取了指針所指的記憶體地址,並訪問該存儲區域的值。

當該存儲器區域不再映射到您的應用時,或者換句話說,該記憶體區域在你認為使用的時候卻沒有使用,該記憶體區域是無法訪問的。 這時內核會拋出一個異常( EXC ),表明你的應用程式不能訪問該存儲器區域(BAD ACCESS) 。

總之,當你碰到EXC_BAD_ACCESS ,這意味著你試圖發送消息到的記憶體塊,但記憶體塊無法執行該消息。但是,在某些情況下, EXC_BAD_ACCESS是由被損壞的指針引起的。每當你的應用程式嘗試引用損壞的指針,一個異常就會被內核拋出。

4.調試EXC_BAD_ACCESS

調試EXC_BAD_ACCESS可能會非常棘手和令人沮喪。然而,現在EXC_BAD_ACCESS不再是一個謎,它沒有想象中的那麼可怕。

你需要知道的第一件事是您的應用程式並不一定是在崩潰的那一刻,無法訪問記憶體區域。這就是常使調試EXC_BAD_ACCESS變得困難的原因。

同樣受損指針也是如此。當你的指針被損壞時,您的應用程式不會崩潰。同時,如果您在應用程式中來回傳遞一個受損的指針也不會崩潰。當應用程式試圖引用受損指針的時候,就會發生奔潰。

5.僵屍調試模式

僵屍調試模式在過去幾年中得到了普及,事實上它們已經出現在Xcode上超過十年。僵屍聽起來有點戲劇性,但它實際上是為幫助我們調試EXC_BAD_ACCESS功能而取得一個偉大的名字。讓我來解釋它是如何工作的。

在Xcode中,您可以啟用僵屍對象,這意味著被釋放的對象將會以僵屍的形式被保留。換言之,保留釋放的對象就是為了調試。這裡沒有涉及任何魔法。如果您向僵屍對象發送消息,你的應用程式將會由於EXC_BAD_ACCESS而崩潰。

這有什麼好處嗎?讓EXC_BAD_ACCESS難以調試的原因是,你不知道你的應用程式試圖訪問哪個對象。僵屍對象在許多情況下解決這個問題。通過保留已釋放的對象,Xcode可以告訴你你試圖訪問哪個對象,這使的查找問題原因容易得多。

在Xcode中啟用僵屍對象是很容易的。註意,這可能會因的Xcode的版本而不同的。以下方法適用於Xcode的6和7,單擊左上角的Edit Scheme,並選中Edit Scheme。

在左側選中Run ,在上方打開 Diagnostics選項。要啟用僵屍對象,勾選 Enable Zombie Objects選框。

figure-edit-scheme.jpg

如果你現在遇到EXC_BAD_ACCESS ,在Xcode的控制台輸出,告訴你該從哪裡查找問題。看看下麵的例子輸出。 

-[ChildViewController respondsToSelector:] message sent to deallocated instance 0x17579780

 

在上面的例子中, Xcode告訴我們, respondsToSelector的消息:被髮送到一個僵屍對象。然而,僵屍對象不再是ChildViewController類的一個實例。以前分配給ChildViewController實例的記憶體區域不再映射到您的應用程式。這為你瞭解問題產生的根本原因提供一個不錯的建議。

不幸的是,僵屍對象將無法保存您的一天每次崩潰的EXC_BAD_ACCESS的記錄。既然僵屍對象沒有這些方法,那麼你可以採取其他的方法進行一些適當的分析。

6.分析

如果僵屍對象不能解決你的問題,那麼問題的根源可能就不那麼簡單了。在這種情況下,您需要仔細看看在應用程式崩潰時執行的代碼。這可能是繁瑣和耗時的。

為了幫助你發現你的代碼的問題,你可以使用Xcode來分析你的代碼,幫助你找到出現問題的地方。註意,Xcode分析項目,它會指出每一個潛在的可能出現的問題的地方。

使用Xcode來分析你的項目,從Xcode的 Product菜單選擇 Analyze或按 Shift-Command-B.Xcode的將需要片刻的時間,但是當它完成的時候你會在左邊的 Issue Navigator看到問題列表。由Analyze發現的問題用藍色高亮顯示。

figure-analyze-1.jpg

當你點擊一個問題,Xcode的會指向問題代碼塊,這些正是你要的註意的地方。註意,Xcode僅僅是建議。在某些情況下,這是可能的,問題是不相關的,不固定。

figure-analyze-2.jpg

如果你找不到造成EXC_BAD_ACCESS的錯誤,那就需要你仔細審視Xcode項目,分析其中發現的每一個問題。

 

7.結論

EXC_BAD_ACCESS是開發者面臨的一個共同的問題,它是手動記憶體管理固有的問題。雖然推行ARC記憶體管理方式 (自動引用計數)使得EXC_BAD_ACCESS沒那麼頻繁,但他們並沒有真正的消失。

http://www.cocoachina.com/articles/15324

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 修改密碼報異常mysql> update user set password=password("y1jhcfzX!") where user="root"; 異常[root@ddtest-mysql01 ~]# mysql -uroot -pEnter password: ERROR 1045 ( ...
  • 摘自: 達夢常用語句 https://blog.csdn.net/zhangxuechao_/article/details/47300953 達夢資料庫管理之表空間 https://blog.csdn.net/lucky_fly/article/details/81566616 --查看表空間se ...
  • [toc] 1、使用比較運算符查詢 MongoDB | 運算 | $gt | 大於 $lt | 小於 $gte | 大於等於 $lte | 小於等於歐 $ne | 不等於 2、使用關鍵字查詢 2.1、in/not in 關鍵字 2.2、size 關鍵字 對於值為list的欄位,可以對list的長度( ...
  • [toc] 1、數據類型 MongoDB常見類型 | 說明 | Object ID|文檔ID String|字元串,最常用,必須是有效的UTF 8 Boolean|存儲一個布爾值,true或false Integer|整數可以是32位或64位,這取決於伺服器 Double|存儲浮點值 Arrays| ...
  • 問題:資料庫內直接操作導致 錯誤 0xc020901c: 數據流任務 1: 源 - yndata1$.輸出[Excel 源輸出] 上的 源 - yndata1$.輸出[Excel 源輸出].列[indications] 出錯。返回的列狀態是:“文本被截斷,或者一個或多個字元在目標代碼頁中沒有匹配項。 ...
  • 這篇文章主要介紹了Oracle排名函數(Rank)實例詳解,需要的朋友可以參考下 --已知:兩種排名方式(分區和不分區):使用和不使用partition --兩種計算方式(連續,不連續),對應函數:dense_rank,rank ·查詢原始數據:學號,姓名,科目名,成績 select * from  ...
  • 本文整理自雲棲社區之前對阿裡搜索事業部資深搜索專家蔣曉偉老師的一次採訪,蔣曉偉老師,認真而嚴謹。在加入阿裡之前,他曾就職於西雅圖的臉書,負責過調度系統,Timeline Infra和Messenger的項目。而後在微軟的SQL Server引擎擔任過Principal Engineer,負責關係數據 ...
  • 前言 開心一刻 閨蜜家暴富,買了一棟大別野,喊我去吃飯,菜挺豐盛的,筷子有些不給力,銀筷子,好重,我說換個竹子的,閨蜜說,這種銀筷子我家總共才五雙,只有貴賓才能用~我咬著牙享受著貴賓待遇,終於,在第三次夾蝦排滑落盤子時,我爆發了:去它喵的貴賓,我要蝦排……不是……我要竹筷子! 連接 簡單來說,就是將 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...