我正在使用NSNotificationCenter试图控制SpriteKit中的计时器。我第一次进入SKScene时代码运行正常,但是当我尝试重新输入SKScene时,我收到了EXC_BAD_ACCESS错误。我认为这与removeObserver函数有关。我不确定何时删除观察者,我试图在prepareForSegue函数中执行此操作但没有成功。我的viewController如下:
class JobStartedViewController: UIViewController {
var titleOfJob: String = ""
override func viewDidLoad() {
super.viewDidLoad()
let skView = self.view as! SKView
let scene:SKScene = GameScene.init(size: skView.bounds.size)
NSNotificationCenter.defaultCenter().postNotificationName("stopTimerNotification", object: nil)
NSNotificationCenter.defaultCenter().postNotificationName("startTimerNotification", object: nil)
/* Sprite Kit applies additional optimizations to improve rendering performance */
skView.ignoresSiblingOrder = true
/* Set the scale mode to scale to fit the window */
scene.scaleMode = .AspectFill
skView.presentScene(scene)
}
我将我的观察者添加到我的GameScene.swift中,如下所示:
class GameScene: SKScene {
override func didMoveToView(view: SKView) {
NSNotificationCenter.defaultCenter().addObserver(self, selector: "stopTimerInBackground:", name:"stopTimerNotification", object: nil)
NSNotificationCenter.defaultCenter().addObserver(self, selector: "startTimerInBackground:", name:"startTimerNotification", object: nil)
答案 0 :(得分:5)
以下是可能的事件流程:
JobStartedViewController
,它会创建场景并将其添加到视图中,触发didMoveToView(_:)
并添加两个观察者。SKView
中删除场景。在不久之后的某个时刻,没有更多强烈的引用场景,它被取消分配。 此时,通知中心仍然存在不安全的引用。 JobStartedViewController
或以其他方式发布stopTimerNotification
通知。NSNotificationCenter
尝试在解除分配的场景上执行选择器并使您的应用崩溃。使用NSNotificationCenter
时的常规做法是在Objective-C的dealloc
方法或Swift的deinit
方法中删除观察者:
class GameScene: SKScene {
// ...
deinit {
NSNotificationCenter.defaultCenter().removeObserver(self)
}
}
如果您计划多次在视图中添加和删除该场景,则还应考虑在willMoveFromView(_:)
中删除观察者。