如何执行链接对象的迁移?



我需要执行迁移,而不会失去父/所有者。如何在迁移块中做到这一点?

迁移之前

class Deck : Object {
@objc dynamic var name : String = ""
var cards = List<Card>()
}
class Card : Object {
@objc dynamic var text = ""
var parentCategory = LinkingObjects(fromType: Deck.self, property: "cards")
}

迁移后

class Deck : Object {
@objc dynamic var name : String = ""
let cards = LinkingObjects(fromType: Card.self, property: "owner")
}
class Card : Object {

@objc dynamic var text = ""
@objc dynamic var owner : Deck?

}

在新版本的应用程序中,当我制作新卡时,我像这样设置父级:

let decks = realm.objects(Deck.self)
owner = decks[index]

我只需要这样做,但是在迁移的上下文中:

let realm = try! Realm()
let decks : Results<Deck>? = realm.objects(Deck.self)
do {
try realm.write {
if decks!.count > 0 {
for indexOfDeck in 0...decks!.count - 1 {
if decks![indexOfDeck].cards.count > 0 {
for indexOfCard in 0...decks![indexOfDeck].cards.count - 1 {
decks![indexOfDeck].cards[indexOfCard].owner = decks![indexOfDeck]
}
}
}
}
}
}catch{
} 

这个问题的挑战在于 LinkingObjects 不是持久化的 - 它们是经过计算的,因此它们不能直接迁移。

第二个挑战是原始卡片对象除了"文本"变量之外没有任何持久属性。

这是将完成这项工作的代码。我们正在获取新的 Decks属性,遍历它们并更新 Card 对象中的每个所有者属性以指向新的套牌。

一旦卡的所有者属性设置为该套牌,该套牌链接卡属性就会"拾取",因为它是经过计算的。

let vers = UInt64(1)
let config = Realm.Configuration( schemaVersion: vers, migrationBlock: { migration, oldSchemaVersion in
print(oldSchemaVersion)
if (oldSchemaVersion < vers) {
print("  performing migration")
migration.enumerateObjects(ofType: Deck.className()) { oldDeck, newDeck in
let cards = newDeck!["cards"] as! List<MigrationObject>
for card in cards {
card["owner"] = newDeck
}
}
} else {
print("no migration needed")
}
})

我的原始对象如下所示,应与问题中的对象匹配

class Deck : Object {
@objc dynamic var name : String = ""
var cards = List<Card>() //current
}
class Card : Object {
@objc dynamic var text = ""
var parentCategory = LinkingObjects(fromType: Deck.self, property: "cards") //current
}

然后更新的对象如下所示

class Deck : Object {
@objc dynamic var name : String = ""
var cards = List<Card>() //current
let linkingCards = LinkingObjects(fromType: Card.self, property: "owner") //new
}
class Card : Object {
@objc dynamic var text = ""
var parentCategory = LinkingObjects(fromType: Deck.self, property: "cards") //current
@objc dynamic var owner : Deck? //new
}

最新更新