来自equatable协议的func ==不适用于自定义对象,swift

时间:2016-07-14 20:59:00

标签: ios swift equatable

我的目标是显示历史记录的用户列表(例如用户名)(如果有)。为了做到这一点,我正在做

1. Create an custom object named User like below

 class User: NSObject
    {
        var login: String

        init(login: String)
        {
            self.login = login
        }
        required init(coder aDecoder: NSCoder) {
            login = aDecoder.decodeObjectForKey("login") as! String
        }

        func encodeWithCoder(aCoder: NSCoder) {
            aCoder.encodeObject(login, forKey: "login")
        }
    }

    // This conform to make sure that I compare the `login` of 2 Users
    func ==(lhs: User, rhs: User) -> Bool
    {
        return lhs.login == rhs.login
    }

在UserManager,我正在saveretrieve User。在保存之前,我正在检查历史记录登录列表是否包含User,我不会添加它,否则。

class UserManager : NSObject
{
    static let sharedInstance   =   UserManager()
    var userDefaults            =   NSUserDefaults.standardUserDefaults()

    func saveUser(user:User)
    {
        var users = retrieveAllUsers()

        // Check before adding
        if !(users.contains(user))
        {
            users.append(user)
        }


        let encodedData =   NSKeyedArchiver.archivedDataWithRootObject(users)
        userDefaults.setObject(encodedData, forKey: "users")
        userDefaults.synchronize()
    }

    func retrieveAllUsers() -> [User]
    {
        guard let data  =   userDefaults.objectForKey("users") as? NSData else
        {
            return [User]()
        }
        let users   =   NSKeyedUnarchiver.unarchiveObjectWithData(data) as! [User]
        // Testing purpose
        for user in users
        {
            print(user.login)
        }
        return users
    }
}

第一次尝试时,我做

UserManager.sharedInstance.saveUser(User(login: "1234"))

现在它保存了第一次登录。第二次,我也做了

UserManager.sharedInstance.saveUser(User(login: "1234"))

UserManager仍会将第二次登录添加到nsuserdefault。这意味着函数contains失败并导致

func ==(lhs: User, rhs: User) -> Bool
{
    return lhs.login == rhs.login
}

无效。

有谁知道为什么或对此有任何想法。

2 个答案:

答案 0 :(得分:11)

问题是User派生自NSObject。这意味着(正如您所说),您的==实施从未被咨询过。对于从NSObject派生的对象,Swift的行为是不同的;它以Objective-C的方式做事。要在从NSObject派生的对象上实现可公平性,请覆盖isEqual:。这就是使NSObject派生的对象在Objective-C和Swift中以自定义方式等同的原因。

只需将此代码粘贴到您的用户类声明中,contains即可按您的意愿开始工作:

override func isEqual(object: AnyObject?) -> Bool {
    if let other = object as? User {
        if other.login == self.login {
            return true
        }
    }
    return false
}

答案 1 :(得分:3)

发生了什么?

正如 @matt 已经说过,问题在于平等。

查找

var users = [User]()
users.append(User(login: "1234"))
users.contains(User(login: "1234")) // false

再看一遍

var users = [User]()
let user = User(login: "1234")
users.append(user)
users.contains(user) // true <---- THIS HAS CHANGED

包含

使用您在此处定义的逻辑

func ==(lhs: User, rhs: User) -> Bool {
    return lhs.login == rhs.login
}

实际上,它只是比较对象的内存地址。

解决方案

您可以解决将自己的逻辑传递给contains的问题,只需替换此

即可
if !(users.contains(user)) {
    users.append(user)
}

用这个

if !(users.contains { $0.login == user.login }) {
    users.append(user)
}