我知道我可以在设置断点或print someFloatVariable
时键入po [self someIvarHoldingAnObject]
,但我无法执行以下有用的操作:
[self setAlpha:1];
然后吐出来:
错误:'[self'不是有效的命令。
奇怪的是我可以打电话给po [self someIvarHoldingAnObject]
,它会打印出它的描述。
我相信我在一年前看过一个视频,其中有人演示了如何在运行时通过控制台执行代码,如果我没有弄错,这个人也提供了参数并为指针分配了对象。怎么做?
答案 0 :(得分:95)
gdb v.lldb命令的规范参考是http://lldb.llvm.org/lldb-gdb.html
您想使用expr命令来计算表达式。它是lldb命令之一,除了参数之外还带有“原始输入”,所以你经常需要一个“ - ”来表示参数(到expr)结束和命令开始的位置。 e.g。
(lldb) expr -- [self setAlpha:1]
有一个快捷方式“p”,为你做(但不允许任何参数),例如。
(lldb) p [self setAlpha:1]
如果您调用的函数不属于您的程序,您通常需要显式声明其返回类型,以便lldb知道如何调用它们。 e.g。
(lldb) p printf("hi\n")
error: 'printf' has unknown return type; cast the call to its declared return type
error: 1 errors parsing expression
(lldb) p (int)printf("hi\n")
(int) $0 = 3
hi
(lldb)
有一种巧妙的解决浮点参数问题的方法,BTW。您创建一个“表达式前缀”文件,该文件将添加到您在lldb中输入的每个表达式中,并带有类方法的原型。例如,我有一个继承自NSObject的类MyClass,它有两个感兴趣的方法,“setArg:”和“getArg”设置并获得一个浮动ivar。这是一个愚蠢的小例子,但它显示了如何使用它。这是我为lldb写的前缀文件:
@interface NSObject
@end
@interface MyClass : NSObject
- init;
- setArg: (float)arg;
- (float) getArg;
@end
extern "C" {
int strcmp (const char *, const char *);
int printf(const char * __restrict, ...);
void puts (const char *);
}
在我添加的~/.lldbinit
文件中
settings set target.expr-prefix /Users/jason/lldb-prefix.h
现在我可以做到
(lldb) p [var getArg]
(float) $0 = 0.5
(lldb) p [var setArg:0.7]
(id) $1 = 0x0000000100104740
(lldb) p [var getArg]
(float) $2 = 0.7
你会注意到我在这里也包含了几个标准的C库函数。执行此操作后,我不再需要再转换这些类型的返回类型,例如
(lldb) p printf("HI\n")
<no result>
HI
(lldb) p strcmp ("HI", "THERE")
(int) $3 = -12
(修复了“&lt; no result&gt;”事件已经提交给lldb TOT来源了。)
答案 1 :(得分:4)
如果您需要多行,请使用expression
:
expression
do {
try thing.save()
} catch {
print(error)
}
// code will execute now
完成并执行代码的空行。