Объект X класса Y не реализует методSignatureForSelector в Swift

У меня есть класс Person, который создается несколько раз. Каждый человек получает свой собственный таймер. В моем init для Person я вызываю startTimer().

class Person {
 var timer = NSTimer()
 func startTimer() {
    timer = NSTimer.scheduledTimerWithTimeInterval(1, target: self, selector: Selector("timerTick"), userInfo: nil, repeats: true)
 }

 func timerTick() {
    angerLevel++
    println("Angry! \(angerLevel)")
 }
...
...
}

Поэтому у меня может быть 3 экземпляра Person в массиве Person[]. Я получаю сообщение об ошибке:

2014-06-25 13:57:14.956 ThisProgram[3842:148856] *** NSForwarding: warning: object 0x113760048 of class '_TtC11ThisProgram6Person' does not implement methodSignatureForSelector: -- trouble ahead

Я читал в другом месте, что я должен наследовать от NSObject, но это в Swift не Obj-C. Функция находится внутри класса, поэтому я не уверен, что делать.

Ответ 1

Не думайте о NSObject как классе Objective-C, думайте об этом как о классе Cocoa/Foundation. Даже если вы используете Swift вместо Objective-C, вы все равно используете все те же структуры.

Два варианта: (1) добавить атрибут dynamic к функции, которую вы хотите использовать в качестве селектора:

    dynamic func timerTick() {
        self.angerLevel++
        print("Angry! \(self.angerLevel)")
    }

Или (2) объявить Person в качестве подкласса NSObject, а затем просто вызвать super.init() в начале вашего инициализатора:

class Person: NSObject {
    var timer = NSTimer()
    var angerLevel = 0

    func startTimer() {
        print("starting timer")
        timer = NSTimer.scheduledTimerWithTimeInterval(1, target: self, selector: "timerTick", userInfo: nil, repeats: true)
    }

    func timerTick() {
        self.angerLevel++
        print("Angry! \(self.angerLevel)")
    }

    override init() {
        super.init()
        self.startTimer()
    }
}

Ответ 2

Так как XCode6 beta 6, вы можете использовать функцию "dynamic"

dynamic func timerTick() { .... }

Ответ 3

У меня была аналогичная ошибка, пытающаяся использовать let encodedArchive = NSKeyedArchiver.archivedDataWithRootObject(archive) as NSData где архив был массивом пользовательского класса. Я обнаружил, что объявить, что пользовательский класс как подкласс NSObject и NSCoding сделал трюк. Для соответствия протоколу NSCoding потребуется еще несколько строк, чтобы он начинал с примерно так:

class Person: NSObject, NSCoding {
  init() {
    super.init()
  }

  func encodeWithCoder(_aCoder: NSCoder) {   }
}