Swift Equatable Protocol


85

Я готував цей підручник для Swift: https://www.raywenderlich.com/125311/make-game-like-candy-crush-spritekit-swift-part-1 і натрапив на цей код:

func == (lhs: Cookie, rhs: Cookie) -> Bool {
    return lhs.column == rhs.column && lhs.row == rhs.row
}

Я писав саме це, але Xcode видає мої помилки:

Consecutive declarations on a line must be separated by ';'
Expected declaration operators are only allowed at global scope

Я знайшов цей код з документації Apple: https://developer.apple.com/documentation/swift/equatable

Що дуже схоже на те, що я писав. Що не так? Це здається мені помилкою. Я використовую Xcode 6 Beta 2

РЕДАГУВАТИ:

Це мій увесь клас Cookie:

class Cookie: Printable, Hashable {
    var column: Int
    var row: Int
    let cookieType: CookieType
    let sprite: SKSpriteNode?
    
    init(column: Int, row: Int, cookieType: CookieType) {
        self.column = column
        self.row = row
        self.cookieType = cookieType
    }
    
    var description: String {
        return "type:\(cookieType) square:(\(column),\(row))"
    }
    
    var hashValue: Int {
        return row * 10 + column
    }
    
    func ==(lhs: Cookie, rhs: Cookie) -> Bool {
        return lhs.column == rhs.column && lhs.row == rhs.row
    }
}

Який код перед цим твердженням? У мене це прекрасно працює само по собі
Коннор,

Я додав цілий клас до опису
Аддісон

8
"Оператори оголошень дозволені лише у глобальному масштабі" Досить похмуро. Це одне з кращих повідомлень про помилки компілятора Swift!
matt

1
Ви можете перевантажити оператор лише у діапазоні файлів .
holex

2
Вам потрібно вийти func ==(lhs: Cookie, rhs: Cookie) -> Bool {...}з класу Cookie !!
Хлунг,

Відповіді:


145

Перемістіть цю функцію

func == (lhs: Cookie, rhs: Cookie) -> Bool {
    return lhs.column == rhs.column && lhs.row == rhs.row
}

Поза класу печива. Це має сенс, оскільки він замінює оператор == у глобальній області, коли він використовується на двох файлах cookie.


3
Я хотів би додати, що на xCode 6.3.2 та swfit 1.2 func == має бути відразу після визначення класу або структури. Навіть додавання простого речення на зразок "var a = 1" поверне помилку компілятора.
fangmobile

2
Я ніколи не подумав би поставити його поза класом! Як це навіть називається? Як знайти його в google?
rr1g0

1
Існує пояснення, чому перевантаження оператора знаходиться в глобальній області , хоча обговорюються можливі зміни, щоб дозволити реалізації оператора всередині типу.

32

SWIFT 2:

Як і в swift 2, NSObjectвже відповідає. EquatableВам не потрібна відповідність у верхній частині, тому це як

class Cookie: NSObject {
    ...

}

І вам потрібно замінити isEqualметод як

class Cookie:NSObject{
    var column: Int
    var row: Int

    //..........

    override func isEqual(object: AnyObject?) -> Bool {
        guard let rhs = object as? Cookie else {
            return false
        }
        let lhs = self

        return lhs.column == rhs.column
    }

}

Цей isEqualметод часу знаходиться всередині класу. :)

EDIT для SWIFT 3: Змініть цей метод як

override func isEqual(_ object: AnyObject?) -> Bool {
        guard let rhs = object as? Cookie else {
            return false
        }
        let lhs = self

        return lhs.column == rhs.column
    }

6

перетворення класу на об'єкт NSO вирішило для мене рівнозначні проблеми ...

class Cookie: NSObject {
...
}

(отримав підказку з підручників для учнів iOS)


1
Це було б тому, що NSObject реалізує наступне в рядку 70 NSObject swiftDoc extension NSObject : Equatable, Hashable.
Адріан Слайтерс
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.