LLDB命令的语法有其通用结构,通常是以下形式的:
<command></command>[<subcommand> [<subcommand>...]] <action> [-options [option-value]] [argument [argument...]]</action></subcommand></subcommand>
其中:
(命令)和(子命令):LLDB调试命令的名称。命令和子命令按层级结构来排列:一个命令对象为跟随其的子命令对象创建一个上下文,子命令又为其子命令创建一个上下文,依此类推。
LLBD命令行的解析操作在执行命令之前完成。上面的这些元素之间通过空格来分割,如果某一元素自身含有空格,则可以使用双引用。而如果元素中又包含双引号,则可以使用反斜杠;或者元素使用单引号。如下所示:
(lldb) command [subcommand] -option "some \"quoted\" string"
(lldb) command [subcommand] -option 'some "quoted" string'
一般开发时候用的最多的情况应该就是打断点,然后打印当前变量的值了吧。
打断点的命令就不说了,反正大多数人都是直接在代码里面添加断点的。基本打印变量的值比较常见的就这几个命令print、po、frame、exp。
- print 该命令可以用来打印基本类型(int之类的),如果打印的是对象,还会打印出对象的指针地址
- po 这个是print object命令的缩写,仅仅是用来打印对象的值,但是用的最多
- exp 该命令不仅有print类似的功能,而且它还能改写对象的值,多数用在测试数据的时候修改传入的参数。
- frame 该命令我用的比较少,知道的仅有frame variable命令可以打印变量的值,以及frame select命令可以打印出当前的代码帧。有兴趣的可以自行查阅
不知道苹果是不是优化了模拟器,以前使用exp self.view.backgroundColor = [UIColor greenColor]
改变背景颜色的时候,还需要加exp (void)[CATransaction flush]
命令才能生效,现在用的Xcode9.3,执行之后直接就改变了。
另外一个常用的命令是image
命令
-
image list
可以打印出工程中使用的库 -
image lookup --address
后面带上崩溃的内存地址,可以具体定位到具体崩溃的代码位置
一些好玩的方法
1、push一个控制器
首先获取导航控制器
exp id $nvc = [[[UIApplication sharedApplication] keyWindow] rootViewController]
然后创建一个新的控制器
exp id $vc = [UIViewController new]
为新的控制器设计背景色
exp (void)[[$vc view] setBackgroundColor:[UIColor yellowColor]]
最后跳转该控制器
exp (void)[$nvc pushViewContoller:$vc animated:YES]
(在Xcode9.3的时候,输入完上述命令继续运行就会直接跳转,如果没有,那么请在后面添加exp (void)[CATransaction flush]
命令吧)
一些问题
不明类型或类型不匹配
p NSLog(@"%@",[self.view viewWithTag:1001])
error: 'NSLog' has unknown return type; cast the call to its declared return type
如果在使用LLDB命令中发现有 unknown type 的类似错误(多见于id类型,比如NSArray中某个值),那我们就必须显式声明类型。比如上面这个命令,我们得这么修改。
p (void)NSLog(@"%@",[self.view viewWithTag:1001])
这样就能得到正确的结果了。 另外,lldb是不支持宏的,需要我们自己替换。
找不到方法
常见于输出frame的时候。比如你可能会得到以下的错误信息:
po self.view.frame
error: unsupported expression with unknown type
这似乎是lldb的一个bug,无法通过点属性访问的方法打印framework里面的对象,但是自己在app里面定义的就可以。我们把上面的命令改动一下:
p (CGRect)[self.view frame]
就可以了