Swift Equatable Протокол

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!
Мэтт
1
вы можете перегрузить оператор только в области видимости файла .
holex
2
Вам нужно выйти func ==(lhs: Cookie, rhs: Cookie) -> Bool {...}из класса Cookie !!
Hlung

Ответы:

145

Переместите эту функцию

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

Вне класса cookie. Это имеет смысл, поскольку он переопределяет оператор == в глобальной области, когда он используется в двух файлах cookie.

Коннор
источник
3
Я хотел бы добавить, что в xCode 6.3.2 и swfit 1.2 функция func == должна стоять сразу после определения класса или структуры. Даже добавление простого предложения типа «var a = 1» вернет ошибку компилятора.
fangmobile
2
Никогда бы не подумал выставить его вне класса! Как это вообще называется? Как мне найти это в гугле?
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метод находится внутри класса. :)

РЕДАКТИРОВАТЬ для 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

создание класса NSObject решило для меня одинаковые проблемы ...

class Cookie: NSObject {
...
}

(получил совет из руководств для учеников iOS)

Фелипе Игнасио Норьега Алькарас
источник
1
Это было бы потому, что NSObject реализует следующее в строке 70 NSObject swiftDoc extension NSObject : Equatable, Hashable.
Адриан Слейтерс,