我已经在标题中添加了下一步,因为这与我上一个问题的问题不同,几乎完全相同。
我有一个Person
实体。
Person
--------
name - mappedKeyName: FullName
email - mappedKeyName: EmailAddress
personID - mappedKeyName: Id
--------
photos
一个Photo
实体。
Photo
--------
image
createDate - mappedKeyName: Date
photoID - mappedKeyName: Id
--------
owner (type Person) - mappedKeyName: UserId - relatedByAttribute: personID
还有其他与Person
相关的对象,而这些对象的JSON就是这样......
{
ObjectId : blah,
Owner : {
Id : 12345asdfg,
FullName : Oliver,
EmailAddress : oliver@oliver.com
}
}
使用此JSON,我的设置适用于导入。创建任何不存在的人(带有Id)。任何确实存在的都会更新。
然而,照片JSON对象就是这样......
{
Id : thisIsThePhotoID,
Date : today,
UserId : 12345asdfg
}
当对象像这样下来时,魔法记录导入在进入人员导入时停止。
代码崩溃在......
- (id) MR_relatedValueForRelationship:(NSRelationshipDescription *)relationshipInfo
{
NSString *lookupKey = [self MR_lookupKeyForRelationship:relationshipInfo];
return lookupKey ? [self valueForKeyPath:lookupKey] : nil; // it stops here.
}
lookupKey
的值是@“personID”。
在断点处打印出relationshipInfo给出了......
$6 = 0x1fd695e0 (<NSRelationshipDescription: 0x1fd695e0>),
name owner,
isOptional 0,
isTransient 0,
entity Photo,
renamingIdentifier owner,
validation predicates (),
warnings (),
versionHashModifier (null)
userInfo {
mappedKeyName = UserId;
relatedByAttribute = personID;
},
destination entity Person,
inverseRelationship photos,
minCount 1,
maxCount 1,
isOrdered 0,
deleteRule 1
我真的不知道为什么这不起作用。我没有得到任何明智的错误报道。
答案 0 :(得分:17)
MagicalRecord无法使用此JSON格式自动映射关系:
{
Id : thisIsThePhotoID,
Date : today,
UserId : 12345asdfg
}
为了让MagicalRecord将关系映射到Person
对象,它也必须是JSON中的对象,例如:
{
Id : thisIsThePhotoID,
Date : today,
User : {
UserId : 12345asdfg
}
}
这样MagicalRecord知道它是一个对象,它将在现有数据库中对具有上述ID的Person
记录进行适当的查找,并映射关系。
但是,这有两个问题。如果您无法更改JSON输出,则必须在Photo
上创建一个类别类,您可以自己手动映射关系。在第二期之后我会谈到这一点。
第二个问题是上述JSON格式假设您已经解析了用户并将记录存储在数据库中。如果你还没有MagicalRecord将使用上面的ID创建一个新的Person
记录,但由于该对象上没有其他属性(注意UserId键是字典中唯一的属性),它将是相当空的并且不包括姓名和电子邮件地址。您可以随时扩展您的JSON(如果您有这种可能性)以在Photo词典中的Person词典中包含这些属性:
{
Id : thisIsThePhotoID,
Date : today,
User : {
UserId : 12345asdfg,
FullName : Oliver,
EmailAddress : oliver@oliver.com
}
}
JSON有效负载非常小,所以如果可以的话,这样做并没有什么坏处。此外,如果数据库中不存在新的Person
记录,它只会创建一个新的Photo
记录。
然后进行手动映射。如果您无法将JSON更改为上述格式,则必须手动覆盖关系映射,因为JSON未按MagicalRecord映射的方式准备。
为名为Photo+Mapping.h/.m
的{{1}}创建一个类别类。我喜欢坚持使用+Mapping
。然后该类应该在标题和实现文件中Photo (Mapping)
,你很高兴。
MagicalRecord有许多可用于覆盖的实例方法(参见MagicalRecord作者撰写的this article on MagicalRecord importing的后半部分),其中包括import<;attributeName>;:
和import<;relationshipName>;:
。类本身上还有willImport:
,didImport:
和shouldImport:
方法,允许您覆盖任何映射。
对于您的情况,您可以使用import<;relationshipName>;:
或shouldImport:
。我拿了这两个因为一个有一点好处,这取决于你是否已经映射了所有Person
个对象,并且它们可用于Photo
对象上的关系映射。
以下是您可以做的事情的例子(如果您愿意,可以选择合并其中一些,这样做也没有坏处)。这里有一个注释:总是在覆盖映射时使用当前NSManagedObjectContext
(通过self.managedObjectContext
可以通过MagicalRecord轻松访问),否则最终会出现上下文问题。
务必导入人物:
#import "Photo+Mapping.h"
#import "Person.h"
// Assuming you only want to import the Photo object if you already have a Person stored this is a great method to tell MagicalRecord whether to continue with importing or not
-(BOOL)shouldImport:(id)data {
Person *person = [Person findFirstByAttribute:data[@"UserId"] value:@"personID" inContext:self.managedObjectContext];
if (!person) {
// no Person object exists so don't import the Photo object - again this is up to you since you might want to create the record if not
return NO;
}
// you can set the relationship here (you might as well) or use the importPerson: method below (doing a second lookup, which is unnecessary at this point)
[self setPerson:person];
return YES;
}
// If you use this method you're doing the lookup to check whether a record exist when MagicalRecord is trying to map the Person relationship
-(void)importPerson:(id)data {
Person *person = [Person findFirstByAttribute:data[@"UserId"] value:@"personID" inContext:self.managedObjectContext];
if (!person) {
// if no Person record exists for the associated UserId, you should create one (or not - if you choose not to, it's wise to throw away this Photo object)
person = [Person createInContext:self.managedObjectContext];
[person setPersonID:data[@"UserId"]];
}
// set the relationship
[self setPerson:person];
}
// finally you can also use the following method when MagicalRecord is done mapping and get rid of the Photo object if the Person relationship is nil:
-(void)didImport:(id)data {
if (!self.person) {
[self deleteInContext:self.managedObjectContext];
}
}
希望这有帮助!如果您有任何问题,请告诉我。