从Rubymotion调用NSSetUncaughtExceptionHandler

时间:2012-09-11 14:38:45

标签: ios rubymotion

我正在尝试使用 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

3 个答案:

答案 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.