Объект 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) { }
}