我的目标是显示历史记录的用户列表(例如用户名)(如果有)。为了做到这一点,我正在做
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,我正在save
和retrieve
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
}
无效。
有谁知道为什么或对此有任何想法。
答案 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)
}