Xcode 調試技巧 --常用命令和斷點

来源:https://www.cnblogs.com/chengxyyh/archive/2020/06/20/13168712.html
-Advertisement-
Play Games

Xcode 中的調試技巧與我們的日常開發息息相關,而這些調試技巧在我們解決Bug時,常常有事半功倍的作用,經常會用到的有各種斷點 和 命令。而這些調試技巧也經常會在面試中問到,所以不知道的就來看看吧。 調試主要觀看區 調試命令 在上圖中,右側綠色區域就是Log 輸出區,在 Log 輸出區可以使用一些 ...


Xcode 中的調試技巧與我們的日常開發息息相關,而這些調試技巧在我們解決Bug時,常常有事半功倍的作用,經常會用到的有各種斷點 和 命令。而這些調試技巧也經常會在面試中問到,所以不知道的就來看看吧。

調試主要觀看區

調試命令

在上圖中,右側綠色區域就是Log 輸出區,在 Log 輸出區可以使用一些命令,來輔助調試。 那有哪些調試命令呢? 想要看所有的調試命令,可以在上圖的右側區域輸入help,就會列出所有的調試命令。 本文就介紹幾個使用頻率比較高的,其他就查看後,自行瞭解吧。

1. p 命令

-- ('expression --') Evaluate an expression on the current thread.
Displays any returned value with LLDB's default formatting.

p 命令是 print 命令的簡寫,使用p 命令可以查看基本數據類型的值,但是如果 使用 p 命令 查看的是對象,那麼只會返回對象的指針地址。 p 命令後面除了可以接 變數、常量,還可以接 表達式。(❌但是不可以使用巨集❌)

2. po 命令

po 命令可以理解為列印對象。功能與 p 命令類似,所以也是可以列印 常量、變數,列印表達式返回的對象等。(❌也不可以列印巨集❌)

p 和 po 使用範例

當然,這些列印功能,除了使用命令外,我們也可以使用左側區域,點擊變數右鍵---> print Description of “xxx”:

Paste_Image.png

當然還有其他的列印方法:

3.expr 命令

expr 是 expression 的簡寫, 使用expr 命令,能夠在調試時,動態的執行賦值表達式,同時列印出結果。我們可以在調試時,動態的修改變數的值,這在調試想要讓應用執行異常路徑(如執行某個else 情況)很有用。

(lldb) p i
(NSInteger) $16 = 1
(lldb) expression i = 5
(NSInteger) $17 = 5
(lldb) po i
5

4.call 命令

上面是動態修改變數的值, Xcode 還支持動態調用函數。在控制台執行該命令,可以在不修改代碼,不重新編譯的情況下,修改界面上的視圖。 這裡有一個動態將cell 的某個子視圖移除的範例:

(lldb) po cell.contentView.subviews
<__NSArrayM 0x60800005f5f0>(
<UILabel: 0x7f91f4f18c90; frame = (5 5; 300 25); text = '2 - Drawing index is top ...'; userInteractionEnabled = NO; tag = 1; layer = <_UILabelLayer: 0x60800009ff40>>,
<UIImageView: 0x7f91f4d20050; frame = (105 20; 85 85); opaque = NO; userInteractionEnabled = NO; tag = 2; layer = <CALayer: 0x60000003ff60>>,
<UIImageView: 0x7f91f4f18f10; frame = (200 20; 85 85); opaque = NO; userInteractionEnabled = NO; tag = 3; layer = <CALayer: 0x608000039860>>
)

(lldb) call [label removeFromSuperview]
(lldb) po cell.contentView.subviews
<__NSArrayM 0x600000246de0>(
<UIImageView: 0x7f91f4d20050; frame = (105 20; 85 85); opaque = NO; userInteractionEnabled = NO; tag = 2; layer = <CALayer: 0x60000003ff60>>,
<UIImageView: 0x7f91f4f18f10; frame = (200 20; 85 85); opaque = NO; userInteractionEnabled = NO; tag = 3; layer = <CALayer: 0x608000039860>>
)

5.bt命令

bt 命令 可以列印出線程的堆棧信息,該信息比左側的Debug Navigator 看到的還要詳細一些。

bt 命令是列印當前線程的堆棧信息

(lldb) bt
* thread #1: tid = 0x27363, 0x000000010d204125 TestDemo-[FifthViewController tableView:cellForRowAtIndexPath:](self=0x00007f91f4e153c0, _cmd="tableView:cellForRowAtIndexPath:", tableView=0x00007f91f5889600, indexPath=0xc000000000400016) + 2757 at FifthViewController.m:91, queue = 'com.apple.main-thread', stop reason = breakpoint 6.1 * frame #0: 0x000000010d204125 TestDemo-[FifthViewController tableView:cellForRowAtIndexPath:](self=0x00007f91f4e153c0, _cmd="tableView:cellForRowAtIndexPath:", tableView=0x00007f91f5889600, indexPath=0xc000000000400016) + 2757 at FifthViewController.m:91
frame #1: 0x0000000111d0a7b5 UIKit-[UITableView _createPreparedCellForGlobalRow:withIndexPath:willDisplay:] + 757 frame #2: 0x0000000111d0aa13 UIKit-[UITableView _createPreparedCellForGlobalRow:willDisplay:] + 74
frame #3: 0x0000000111cde47d UIKit-[UITableView _updateVisibleCellsNow:isRecursive:] + 3295 frame #4: 0x0000000111d13d95 UIKit-[UITableView _performWithCachedTraitCollection:] + 110
frame #5: 0x0000000111cfa5ef UIKit-[UITableView layoutSubviews] + 222 frame #6: 0x0000000111c61f50 UIKit-[UIView(CALayerDelegate) layoutSublayersOfLayer:] + 1237
frame #7: 0x00000001117a5cc4 QuartzCore-[CALayer layoutSublayers] + 146 frame #8: 0x0000000111799788 QuartzCoreCA::Layer::layout_if_needed(CA::Transaction) + 366
frame #9: 0x0000000111799606 QuartzCoreCA::Layer::layout_and_display_if_needed(CA::Transaction*) + 24 frame #10: 0x0000000111727680 QuartzCoreCA::Context::commit_transaction(CA::Transaction
) + 280
frame #11: 0x0000000111754767 QuartzCoreCA::Transaction::commit() + 475 frame #12: 0x00000001117550d7 QuartzCoreCA::Transaction::observer_callback(__CFRunLoopObserver, unsigned long, void) + 113
frame #13: 0x0000000110743e17 CoreFoundation__CFRUNLOOP_IS_CALLING_OUT_TO_AN_OBSERVER_CALLBACK_FUNCTION__ + 23 frame #14: 0x0000000110743d87 CoreFoundation__CFRunLoopDoObservers + 391
frame #15: 0x0000000110728b9e CoreFoundation__CFRunLoopRun + 1198 frame #16: 0x0000000110728494 CoreFoundationCFRunLoopRunSpecific + 420
frame #17: 0x0000000114390a6f GraphicsServicesGSEventRunModal + 161 frame #18: 0x0000000111b9d964 UIKitUIApplicationMain + 159
frame #19: 0x000000010d21294f TestDemomain(argc=1, argv=0x00007fff529fe620) + 111 at main.m:14 frame #20: 0x000000011458a68d libdyld.dylibstart + 1
(lldb)

bt all 命令是列印所有線程的堆棧信息。列印出來的信息太多,就不展示了!

作為一個開發者,有一個學習的氛圍跟一個交流圈子特別重要,這是一個我的iOS交流群:519832104 不管你是小白還是大牛歡迎入駐,分享經驗,討論技術,大家一起交流學習成長!

另附上一份各好友收集的大廠面試題,需要iOS開發學習資料、面試真題,可以添加iOS開發進階交流群,進群可自行下載!

6.image 命令

image list 命令可以列出當前App中的所有module(這個module 在後面符號斷點時有用到),可以查看某一個地址對應的代碼位置。 除了 image list 還有 image addimage lookup等命令,可以自行查看。 當遇到crash 時,查看線程棧,只能看到棧幀的地址,使用 image lookup –address 地址 可以方便的定位到這個地址對應的代碼行。

斷點

Xcode 中的斷點也是很有學問的,有普通斷點、條件斷點、符號斷點、異常斷點等很多種。

1.普通斷點

打一個普通斷點,只需要找到對應的行,在代碼左側(行號上)點擊一下即可。

2.條件斷點

條件斷點是一種很有用的斷點,特別是在for 迴圈中。如果我們需要在i = 5 時添加斷點,其他時候不加,那麼就可以使用條件斷點。條件斷點是在普通斷點上 右鍵,選擇 Edit Breakpoint...,再設置一個條件即可

編輯普通斷點

添加條件

3.符號斷點

符號斷點就是 Symbolic Breakpoint,其實是針對某一個特定函數的斷點,可以是一個 OC函數,也可以是 C++函數。 添加的地方如下:

符號斷點

符號斷點條件

Symbol 欄 可以填 [類名 方法名]或者 方法名 ,module 也是選填項,它就是上面 image 命令中列出來的module。 例如 ,我們如果只填一個viewDidLoad,那麼就會在所有類(包括第三方庫)的viewDidLoad 處打斷點。

符號斷點在調試一些沒有源碼的模塊時比較有用,比如調試一個第三方提供的Lib庫,或者系統的模塊,可以在相應函數處下斷點,可以大概調試清楚程式的運行流程,也可以在斷點的時候查看到參數信息。

4.異常斷點

如果程式運行就崩潰,我們可以打一個異常斷點,這樣崩潰時就會觸發斷點,很容易定位到問題所在,也能看到更多的崩潰相關信息,如Log,函數調用棧。

異常斷點

可以修改異常斷點的條件

註意: 有的程式或者有的功能可能會使用異常來組織程式邏輯,比如調用AVAudioPlayer ,運行到 AVAudioPlayer 時,就會導致斷點被觸發。我們可以修改 Exception 參數,或者取消掉異常斷點來解決。

5.Watch 斷點

當某個變數發生變化的時候會觸發。 創建一個Watch斷點:

關於 Xcode 調試技巧中的 斷點和命令就先這麼多了,其他有用到的以後再補充。

點擊此處,立即與iOS大牛交流學習


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

-Advertisement-
Play Games
更多相關文章
  • 一 CLI訪問OpenShift資源 1.1 資源操作 OCP將OpenShift集群中的為由主節點管理的對象統稱為資源,如:node、service、pod、project、deployment、user。 即使針對的是不同的資源,OpenShift命令行工具也提供了一種統一的、一致的方法來更新、 ...
  • ffmpeg -i test.mp4 -codec copy -bsf: h264_mp4toannexb -f h264 test.264 從MP4文件內提取視頻流,忽略音頻流,指定幀頻、碼率 ffmpeg -i test.mp4 -vcodec h264 -an -r 25 -b:v 256k ...
  • 初用MySQL Mysql示例庫 Navicat15 查看初始密碼 MySQl首次啟動會創建“超級管理員賬號”root@localhost,初始密碼存儲在日誌文件中,通過grep搜索並查看: grep 'temporary password' /var/log/mysqld.log 進入mysql ...
  • 對於MySQL的一些個規範,某些公司建表規範中有一項要求是所有欄位非空,意味著沒有值的時候存儲一個預設值。其實所有欄位非空這麼說應該是絕對了,應該說是儘可能非空,某些情況下不可能給出一個預設值。那麼這條要求,是基於哪些考慮因素,存儲空間?相關增刪查改操作的性能?亦或是其他考慮?該理論到底有沒有道理或 ...
  • 1. 安裝資料庫 1) yum -y install mysql-server(簡單) yum命令自動從網上尋找mysql服務資源,下載至本地並完成安裝 2) 也可以自己在網上下載mysql服務,通過xftp傳輸至Linux系統,自己安裝(一般安裝在usr或opt目錄下) 2. 啟動資料庫 安裝完畢 ...
  • MySQL中給一張千萬甚至更大量級的表添加欄位一直是比較頭疼的問題,遇到此情況通常該如果處理?本文通過常見的三種場景進行案例說明。 1、 環境準備 資料庫版本: 5.7.25-28(Percona 分支) 伺服器配置: 3台centos 7虛擬機,配置均為2CPU 2G記憶體 資料庫架構: 1主2從的 ...
  • 一、我們創建一個新的android項目來進行演示廣播機制中是如何​顯示網路狀態的。 package com.example.broadcasttest2; import android.app.Activity; import android.content.BroadcastReceiver; i ...
  • 一、block記憶體管理 1.block記憶體類型 block記憶體分為三種類型: _NSConcreteGlobalBlock(全局) _NSConcreteStackBlock(棧) _NSConcreteMallocBlock(堆) 2.三種類型的記憶體的創建時機 1)對於_NSConcreteSta ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...