我正在尝试使用 Rubymotion 应用中的NSSetUncaughtExceptionHandler
连接全局异常处理程序。我无法弄清楚如何做到这一点,可能是我使用Rubymotion函数指针语法不正确。
在application(application, didFinishLaunchingWithOptions:launchOptions)
我有:
NSSetUncaughtExceptionHandler nil do |exception|
NSLog "exception handler block"
applicationFailedWithException(exception)
end
处理程序(从未调用过):
def applicationFailedWithException(exception)
NSLog "applicationFailedWithException"
end
然后,在运行时,当某些NoMethodError发生在异常处理程序块的某处时,都不会调用处理程序方法。我在模拟器和设备上都尝试过 - 没有成功。
P.S。有没有办法在崩溃日志中获取ruby方法堆栈,崩溃日志中的最后一个应用程序方法始终是rb_rb2oc_exc_handler
而不是特定于应用程序的方法,这很奇怪。
由于
更新2015
似乎使用较新版本的Rubymotion,您必须在实例变量中保存对处理程序的引用,否则它将无法工作。所以看起来应该是这样的:
def application(application, didFinishLaunchingWithOptions:launchOptions)
@exceptionHandler = lambda { |exception| uncaughtExceptionHandler(exception) }
# or you can use a method reference, which is the same
# @exceptionHandler = method :uncaughtExceptionHandler
NSSetUncaughtExceptionHandler @exceptionHandler
...
end
def uncaughtExceptionHandler(exception)
# log it or send it to crashlytics / flurry / whatever
NSLog "Fatal exception catched\nName: #{exception.name}\nReason: # {exception.reason}\nInfo#{exception.userInfo}"
end
答案 0 :(得分:1)
为了使其工作,我必须为变量分配一个块,然后使用该变量:
handler = lambda do |exception|
NSLog "Exception Name: #{exception.name}"
end
NSSetUncaughtExceptionHandler(handler)
如果我在REPL中输入NSException.raise("test", format: "test")
之类的东西,我会看到NSLog做的事情:
2012-09-11 09:33:05.949 exc[14723:c07] Exception Name: test
2012-09-11 09:33:05.950 exc[14723:c07] *** Terminating app due to uncaught exception 'test', reason: 'test'
*** First throw call stack:
(0x1647022 0x22bcd6 0x15efa48 0x15ef9b9 0x8f00743 0xe3c0c 0x8f0031a 0x8f000ad 0xd4c9fa2 0x259e8d9 0x259f509 0x157e803 0x157dd84 0x157dc9b 0x261f7d8 0x261f88a 0x563626 0x25d0 0x23b5)
terminate called throwing an exception%
理论上,您应该能够将lambda直接嵌套为参数,而不是先将其分配给变量,但我不认为RubyMotion会正确保留该块,如果您尝试它,应用程序会崩溃而不会出错。
答案 1 :(得分:1)
迪伦的答案很棒。我很喜欢传递像lambdas这样的方法:
def application(application, didFinishLaunchingWithOptions:launchOptions)
NSSetUncaughtExceptionHandler method(:on_exception)
end
def on_exception(exception)
puts "Exception Name: #{exception.name}"
end
答案 2 :(得分:0)
根据Apple documentation,您必须在项目中添加ExceptionHandling.framework
。但即便如此:
Rakefile
Motion::Project::App.setup do |app|
app.frameworks += %w(ExceptionHandling)
end
它不适用于OS X.