前言
本Session
主要内容为了解如何使用 LLDB进行应用的调试;了解如何分析堆栈跟踪和诊断Bug
。以及如何利用Swift REPL(Read-Eval-Print Loop)
进行自己代码的测试和应用的探究.
内容
使用LLDB
What is LLDB
LLDB
是一款开源的高性能(debugger)
调试器,也是Xcode
上默认的调试器,支持对Mac应用上的C,Objective-C,C++,Swift以及iOS设备的调试.
Why to use
-
App开发中经常遇到App跑起来却莫名
Crash
或者执行某个操作使得App直接Crash
退出的情况,有时候系统能明显地提示我们导致应用Crash
的代码所在位置,而有时却只是简单地在Main
文件中显示应用Crash
对代码的定位,结合LLDB
调试器的命令就能帮助我们面对一个Crash的应用,更快更清楚地了解到问题代码的位置和问题原因了.
面对正常运行的应用出现了数据读写错误或者UI界面异常的情况时,使用
LLDB
调试器的命令集,能将应用代码任意指定位置进行主动的暂停,获取当前代码所在堆栈的信息,输出一些局部变量值或者堆中的变量值,让我们能更好的理解应用的运行过程,并更细节地了解所出问题的原因.
总的来说,LLDB就是用来帮助修复已存在代码所出
Bug
的工具.
How to use
一. 面对Crash的应用
-
当应用
Crash
时,Xcode
就自动进入了LLDB调试模式,就可以在Debug 命令栏输入特定的LLDB
命令,与LLDB
交互了
-
首先根据系统给出Crash信息了解其Crash的原因,查找Crash的大致位置
然后使用LLDB命令集查看代码的执行情况,以及问题代码的位置,进行对变量值进行查看.
以下为查看Crash原因时常用的LLDB命令集中:ti
:thread info
的缩写,会对应输出应用当前所用到所有线程的具体信息,会给予stop reason
.bt
: (thread backtrace
),会对应输出指定线程的出现Crash
时正在执行的所有函数调用帧,调用层次由下到上递增,在所给的信息中寻找top_level_code
和相关的类文件信息.f number
: (frame select number
),就会进入指定number
的函数栈,简单输出该函数栈的上下文信息.p variableName
: (expression variableName
),对执行变量名内部值进行输出查看.
二. 对数据UI异常的应用
- 通过Xcode快捷的断点添加或者使用命令都能添加
Breakpoint
对运行的应用进行主动的停止.-
b fileName.swift:N
:(breakpoint set --file fileName.swift --line N)
表示为fileName文件的第N行的代码添加断点 -
br l
:(breakpoint list)
输出应用当前所有的断点 -
br dis N
:(breakpoint disable N)
让指定N序号的断点失效 -
br s -r Expression
:(breakpoint set --func-regex Expression)
结合正则表达式匹配目标代码,进行断点设置. (也使用-r ClassName\\.
或者-r main\\.
来匹配是否为组件的函数还是类的方法) -
br m -c "if ture"
:(breakpoint modify --condition "if true")
对断点代码进行条件监听当条件满足时应用才会停止. -
br co a
:(breakpoint command add)
,进入debugger
命令模式,只有输入DNOE
(必须大写)表示命令结束. -
c(continue)
和run
: 跳过该断点让应用继续运行.
-
REPL的调试工作流
What's the REPL
全称为Read-Eval-Print Loop
(“读取-求值-输出”循环), 一个简单的,交互式的编程环境,用来接收代码的输入并直接输出回应.
Why use LLDB REPL
在LLDB命令行中使用LLDB REPL,可以用来对已存在代码进行测试验证,也可以即时添加新的代码进行使用.
How to use LLDB REPL
启动LLDB REPL
方法一: 在终端,输入xcrun swift
,加载REPL不关联任何程序
方法二: 在Xcode中主动断点应用,输入repl
,加载与当前target
相关联的REPL,输入:
返回到LLDB模式.在REPL模式下使用
:
+ LLDB命令会像常规的LLDB调试模式下一样输出命令结果-
在REPL模式下,可以在调试命令行上调用已实现的函数或者添加新的函数并调用.(没有代码提示😔)
Note: 在REPL模式下添加代码的作用域为全局,其生成的变量也是全局变量
结尾
学会使用LLDB命令,结合LLDB和RELP在Swift应用能进行高效的Debug工作和帮助我们更加理解自己应用内部具体的实现.关于LLDB和RELP如何调试应用的技巧还有很多很多,更多的是需要自己真正在项目中实践和学习总结.