1

我有一个相当简单直接的问题,我试图用 Realm 解决。我有在它们(线程)上具有数组属性的对象。当我通过我们的 API 获取所有线程时,它们都被持久化到 Realm 中,因为父对象被单独保存,因此数组属性中的所有子对象(消息和用户)也被正确持久化。但是在应用程序的生命周期中,我需要将新消息添加到该数组属性中。这是我正在尝试做的事情:

func addPubNubMessageToThread(notification: NSNotification) {
    if let info = notification.userInfo as? Dictionary<String, AnyObject> {

        var embeddedMessage = Message(json: (info["data"] as? NSDictionary)!)

        let threadId = (info["thread"]! as String)

        // Persist the message to Realm for future use
        var respectiveThread = Thread(forPrimaryKey: threadId)
        let realm = RLMRealm.defaultRealm()
        realm.beginWriteTransaction()
        respectiveThread.conversation.insertObject(embeddedMessage, atIndex: UInt(0)) // Always fails here in XCode with the error below
        realm.addOrUpdateObject(respectiveThread)
        realm.commitWriteTransaction()
    }
}

但每次我收到以下错误:

 *** Terminating app due to uncaught exception 'RLMException', reason: 'Setting unique property '_id' with existing value '540729b543dd5d1868a42b5d''

有关更多上下文,这是我的 Realm 模型:

class Message: RLMObject {
    dynamic var _id = ""
    dynamic var type = ""
    dynamic var text = ""
    dynamic var author = User()
    dynamic var created = NSDate()
    dynamic var lastUpdated = NSDate()
}

class Thread: RLMObject {
    dynamic var _id = ""
    dynamic var name = ""
    dynamic var conversation = RLMArray(objectClassName: Message.className())
    dynamic var participants = RLMArray(objectClassName: User.className())
    dynamic var created = NSDate()
    dynamic var lastUpdated = NSDate()
}

class User: RLMObject {
    dynamic var _id = ""
    dynamic var name = ""
    dynamic var firstName = ""
    dynamic var lastName = ""
    dynamic var email = ""
    dynamic var phone = ""
    dynamic var username = ""
    dynamic var avatar = NSData()
    dynamic var created = NSDate()
    dynamic var lastUpdated = NSDate()
}

每条消息都有一个名为 author 的属性,_id它抱怨的是_id消息的作者(或用户对象)。错误信息很难解读。我认为这是说我正在尝试使用已经存在的主键创建一个新的用户对象。如果这是问题所在,我应该怎么做才能将新的 Realm 对象添加到已经持久化对象的数组属性中?

编辑

我正在为每个模型设置主键,如下所示:

override class func primaryKey() -> String {
    return "_id"
}

并且_id是由 MongoDB 生成的 GUID……所以它是全球唯一的。

4

2 回答 2

1

问题是,当调用 insertObject 您的对象时,所有子对象都是在 Realm 中创建的,而不是在它们已经存在时更新。如果您首先明确更新您的对象(这也适用于所有子对象),那么这应该可以避免这个问题:

var persistedMessage = realm.addOrUpdateObject(embeddedMessage)
respectiveThread.conversation.insertObject(persistedMessage, atIndex: UInt(0))
于 2014-10-14T22:03:58.870 回答
0

您的问题可能出在您的Message(json:)初始化程序中。当您处理消息的 JSON 时,我猜测作者 ID 已传递(可能还有其余的作者数据)。在您的初始化程序中,您可能正在实例化一个新用户,而不是获取领域中现有用户的实例。我重新创建了您的模型对象,然后创建了现有用户、线程和消息。我能够反序列化来自 JSON 的消息并将新消息添加到现有对话数组中,只要我的反序列化逻辑抓住了现有用户(如果提供的话)。这是我的示例初始化程序:

init(json : NSDictionary) {
    self._id = json["_id"] as String
    self.type = json["type"] as? String ?? ""
    self.text = json["text"] as? String ?? ""
    self.created = json["created"] as? NSDate ?? NSDate()
    self.lastUpdated = json["lastUpdated"] as? NSDate ?? NSDate()

    if let author = json["author"] as? NSDictionary {
        if let authorId = author["_id"] as? String {
            self.author = User(forPrimaryKey: authorId)
        }
    }

    super.init()
}

在这种情况下,我的示例 JSON:

{
    "_id": "2",
    "text": "new message", 
    "author": {"_id": "1"}
}

最后,读取 JSON 并添加到已保存线程的代码:

var error: NSError?
var jsonDictionary : NSDictionary!
if let dictionary = NSJSONSerialization.JSONObjectWithData(jsonData, options: NSJSONReadingOptions.allZeros, error: &error) as? NSDictionary {
    jsonDictionary = dictionary
} else {
    NSLog("\(error)")
    return
}

let realm = RLMRealm.defaultRealm()

realm.beginWriteTransaction()

let newMessage = Message(json: jsonDictionary)
self.thread.conversation.addObject(newMessage)
realm.addOrUpdateObject(self.thread)

realm.commitWriteTransaction()
于 2014-10-23T17:48:16.473 回答