RestKit 创建多对多关系

Posted

技术标签:

【中文标题】RestKit 创建多对多关系【英文标题】:RestKit creating multiple to many relationships 【发布时间】:2013-07-01 19:55:50 【问题描述】:

我正在尝试解析 json 响应以用于核心数据模型。这是来自 json 的示例:

  
      "@context": "TVSchedule",
      "ReturnCode": "0",
      "ReturnMessage": "Successful request",
      "Channel": [
        
          "ChannelId": "http%…..0",
          "Program": [
            
              "@programId": "http……..",
              "Title": "Divorce Court",
              "ProgramLogo": "http://00_180x101.png",
              "ProgramLogos": [
                
                  "@size": "small",
                  "#text": "http://.png"
                ,
                
                  "@size": "large",
                  "#text": "191.png"
                
              ],
              "ProgramDetailsURL": "http:9",
              "PublishedStartTime": "2013-07-01T19:00:00",
              "PublishedEndTime": "2013-07-01T19:30:00",
              "Duration": "00:00:30:00",
              "RatingInfo": 
                "@system": "MPAA",
                "@code": "TV-PG",
                "@age": "10",
                "Title": "Not recommended for children under 10 years",
                "Logo": "http://"
              ,
              "ShortDescription": "She says she cannot trust him .",
              "Year": "2013",
              "Genres": [
                "Series",
                "Reality",
                "Public Affairs",
                "News",
                "Episodic"
              ]
            ,

我需要在 CoreData 中获取它,其中我有一个名为 Channel 的实体,它将持有 channelId 与名为 program 的实体的一对多关系,该实体将保留该级别的属性列表,并且必须- 与实体 ProgramLogo 的许多关系(来自 json 文件中的“ProgramLogos”) Genres(流派字符串数组将添加为仅包含一个字符串属性的多个 Genre 实体),RatingsInfo 实体(来自对应的一对一关系字典),RatingsInfo 不会出现在所有程序中......

这是我使用的 RestKit 代码:

 RKEntityMapping *channelMapping = [RKEntityMapping mappingForEntityForName:kCDChannelEntity inManagedObjectStore:managedObjectStore];
    channelMapping.identificationAttributes = @[ kCDChannelId ];

    [channelMapping addAttributeMappingsFromDictionary:@
                                       kJsonChannelId : kCDChannelId
     ];



    RKEntityMapping *programMapping = [RKEntityMapping mappingForEntityForName:kCDProgramEntity inManagedObjectStore:managedObjectStore];


    [programMapping addAttributeMappingsFromDictionary:@
                                       @"@programId" : kCDProgramId ,
                                     @"ProgramLogo" : kCDProgramLogo,
                              @"ProgramDetailsURL" : kCDProgramDetailsUrl,
                                       @"Duration"  : kCDProgramDuration,
                              @"PublishedStartTime" : kCDProgramStartTime,
                                @"PublishedEndTime" : kCDProgramEndTime,
                                           @"Title" : kCDProgramTitle,
                                            @"Year" : kCDProgramYear,
                                @"ShortDescription" : kCDProgramShortDescription
     ];

    [RKObjectMapping addDefaultDateFormatterForString:@"yyyy'-'MM'-'dd'T'HH':'mm':'ss'Z'" inTimeZone:nil];

    [channelMapping addPropertyMapping:[RKRelationshipMapping relationshipMappingFromKeyPath:@"Program" toKeyPath:kCDChannelHasProgramsRel withMapping:programMapping]];// ??


    RKEntityMapping *parentalRatingMapping = [RKEntityMapping mappingForEntityForName:kCDParentalRatingEntity inManagedObjectStore:managedObjectStore];
    [parentalRatingMapping addAttributeMappingsFromDictionary:@
                                           @"Program.RatingsInfo.@age": kCDParentalRatingAge,
                                         @"Program.RatingsInfo.@code" : kCDParentalRatingCode,
                                         @"Program.RatingsInfo.Logo" : kCDParentalRatingLogo,
                                       @"Program.RatingsInfo.@system" : kCDParentalRatingSystem,
                                        @"Program.RatingsInfo.Title" : kCDParentalRatingTitle];

    [programMapping addPropertyMapping:[RKRelationshipMapping relationshipMappingFromKeyPath:@"Program.RatingsInfo" toKeyPath:kCDProgramHasParentalRatingRel withMapping:parentalRatingMapping]];// ??



    RKEntityMapping *genresMapping = [RKEntityMapping mappingForEntityForName:kCDGenreEntity inManagedObjectStore:managedObjectStore];
    [genresMapping addAttributeMappingsFromDictionary:@
     @"Program.Genres": kCDGenreName
     ];

    [programMapping addPropertyMapping:[RKRelationshipMapping relationshipMappingFromKeyPath:@"Program.Genres" toKeyPath:kCDProgramHasGenres withMapping:genresMapping]];


    RKEntityMapping *logoMapping = [RKEntityMapping mappingForEntityForName:kCDProgramLogoEntity inManagedObjectStore:managedObjectStore];
    [logoMapping addAttributeMappingsFromDictionary:@
     @"Program.ProgramLogos.@size" : kCDLogoSize,
     @"Program.ProgramLogos.#text" : kCDLogoText
     ];

好的,现在有两个问题: 1)如何解析两个非KVO数组(流派,不同实体中的每个字符串,以及ProgramLogos(实体中的每个dict)

2) RestKit 不解析 programId (key "@programId") 为什么?键中的“@”会停止解析吗?

我明白了

*** Terminating app due to uncaught exception 'NSUnknownKeyException', reason: '[<__NSCFDictionary 0x958f980> valueForUndefinedKey:]: this class is not key value coding-compliant for the key programId.'

【问题讨论】:

【参考方案1】:

看起来您的channelResponseDescriptor 应该有一个@"Channel" 的keyPath。由于所有常量,很难判断是否还有其他问题。似乎只有***描述符需要更多关于它应该应用在哪里的信息。


通常,您创建的所有映射都相互关联。根据您的 JSON 结构,您很可能不需要这么多响应描述符,只需要通道的***描述符,然后所有映射都用于使用关系向下导航结构。


例如,你有 JSON:

  "Channel": [
    
      "ChannelId": "http%…..0",
      "Program": [
        ...

您可以在频道和节目实体之间建立关系,并在映射之间建立对应关系,这样当频道被映射时,它将导航到 JSON 的节目部分,映射节目,然后连接对象。你似乎有这个:

[channelMapping addPropertyMapping:[RKRelationshipMapping relationshipMappingFromKeyPath:kJsonProgram toKeyPath:kCDChannelHasProgramsRel withMapping:programMapping]];

(您的代码中有错字,因为它应该应用于channelMapping,而不是programMapping

所以programResponseDescriptor 可能不是必需的,可能只会导致您的映射结果包含重复项。

【讨论】:

我也是这么想的......我不确定keyPaths......它们是如何为嵌入的对象和关系创建的 你能举个例子吗? 没有任何实体需要它自己的响应描述符吗? 您需要至少一个通道的响应描述符。 我删除了其他响应描述符,现在通道得到处理,但没有其他任何东西......

以上是关于RestKit 创建多对多关系的主要内容,如果未能解决你的问题,请参考以下文章

RestKit 多对多关系在连接表中保存新行,在主表中保存空值

RestKit 与 Core Data 多对多:它有效,但我做得对吗?

Hibernate的多对多关联关系

第六章创建多对多关系的三种方式

如何在后端创建多对多关系

15)django-ORM(多对多)