11

我正在使用RestKit,我想解析元素并将其保存到核心数据。我有两个 json 文件:

第一(类别):

[
   {
     "cat_id": 3371,
     "cat_name": "myName",
     "image": 762
   },
   {
     "cat_id": 3367,
     "cat_name": "anotherName",
     "image": 617
   }
]

第二个(元素):

[
  {
    "art_id": "1",
    "node": {
      "author": "name"
    },
    "small_url": 0
  },
  {
    "art_id": "12",
    "node": {
      "author": "anotherName"
    },
    "small_url": 0
  }
]

所以基本思想是每个类别里面都有一些元素。所以这是我的 CoreData 结构: 在此处输入图像描述

我已经下载了 restkit 示例并使用 TwitterCoreData 示例。我的代码是: AppDelegeta.m

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
    NSURL *baseURL = [NSURL URLWithString:@"http://globalURL.com"];
    RKObjectManager *objectManager = [RKObjectManager managerWithBaseURL:baseURL];
    NSManagedObjectModel *managedObjectModel = [NSManagedObjectModel mergedModelFromBundles:nil];
    RKManagedObjectStore *managedObjectStore = [[RKManagedObjectStore alloc] initWithManagedObjectModel:managedObjectModel];
    objectManager.managedObjectStore = managedObjectStore;

    RKEntityMapping *categoryMapping = [RKEntityMapping mappingForEntityForName:@"Category" inManagedObjectStore:managedObjectStore];
    categoryMapping.identificationAttributes = @[ @"catId" ];
    [categoryMapping addAttributeMappingsFromDictionary:@{
     @"cat_id": @"catId",
     @"node.author": @"author",
     }];

    RKEntityMapping *elementsMapping = [RKEntityMapping mappingForEntityForName:@"Elements" inManagedObjectStore:managedObjectStore];
    elementsMapping.identificationAttributes = @[ @"artId" ];
    [elementsMapping addAttributeMappingsFromDictionary:@{
     @"art_id": @"artId",
     @"node.author": @"author",
    }];
    [elementsMapping addPropertyMapping:[RKRelationshipMapping relationshipMappingFromKeyPath:@"category" toKeyPath:@"category" withMapping:categoryMapping]];
    RKResponseDescriptor *responseDescriptor = [RKResponseDescriptor responseDescriptorWithMapping:elementsMapping
                                                                                       pathPattern:nil
                                                                                           keyPath:nil
                                                                                       statusCodes:RKStatusCodeIndexSetForClass(RKStatusCodeClassSuccessful)];
    [objectManager addResponseDescriptor:responseDescriptor];


    [managedObjectStore createPersistentStoreCoordinator];
    NSString *storePath = [RKApplicationDataDirectory() stringByAppendingPathComponent:@"MyCoreData.sqlite"];
    NSString *seedPath = [[NSBundle mainBundle] pathForResource:@"MyCoreData" ofType:@"sqlite"];
    NSError *error;
    NSPersistentStore *persistentStore = [managedObjectStore addSQLitePersistentStoreAtPath:storePath fromSeedDatabaseAtPath:seedPath withConfiguration:nil options:nil error:&error];
    NSAssert(persistentStore, @"Failed to add persistent store with error: %@", error);

    // Create the managed object contexts
    [managedObjectStore createManagedObjectContexts];

    // Configure a managed object cache to ensure we do not create duplicate objects
    managedObjectStore.managedObjectCache = [[RKInMemoryManagedObjectCache alloc] initWithManagedObjectContext:managedObjectStore.persistentStoreManagedObjectContext];



    self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
    // Override point for customization after application launch.
    self.viewController = [[ViewController alloc] initWithNibName:@"ViewController" bundle:nil];
    self.window.rootViewController = self.viewController;
    [self.window makeKeyAndVisible];
    return YES;
}

ViewController.m

NSFetchRequest *fetchRequest = [NSFetchRequest fetchRequestWithEntityName:@"Elements"];
NSSortDescriptor *descriptor = [NSSortDescriptor sortDescriptorWithKey:@"artId" ascending:NO];
fetchRequest.sortDescriptors = @[descriptor];
[[RKObjectManager sharedManager] getObjectsAtPath:@"/detailaddress/:catId" parameters:nil success:^(RKObjectRequestOperation *operation, RKMappingResult *mappingResult) {
    RKLogInfo(@"Load complete: Table should refresh...");
    NSLog(@"%@",mappingResult);
    [[NSUserDefaults standardUserDefaults] setObject:[NSDate date] forKey:@"LastUpdatedAt"];
    [[NSUserDefaults standardUserDefaults] synchronize];
} failure:^(RKObjectRequestOperation *operation, NSError *error) {
    RKLogError(@"Load failed with error: %@", error);
}];

映射日志向我显示“nil”。如何使用restkit将我的第一个json(类别)中的数据保存到核心数据中?请记住,我还没有元素列表。

当我使用创建新文件来创建 NEManagedObject 子类时,我有Elements类。

@interface Elements : NSManagedObject

@property (nonatomic, retain) NSNumber * artId;
@property (nonatomic, retain) NSString * author;
@property (nonatomic, retain) NSString * title;
@property (nonatomic, retain) NSManagedObject *category;

@end
4

2 回答 2

41

在几个月前的 .20 版本发布之前,我一直使用 RESTKit。老实说,虽然这个图书馆有很好的意图,但根据我的经验,它最终浪费的时间比它用来拯救我的时间要多得多。

我已经将它与使用 .NET/MVC、PHP、python/Django/TastyPie 创建的 API 一起使用,并破解了它以使用 Twitter 做一些事情。这一切最终成为一个非常有趣的学术活动,但最终仅此而已。RESTKit 的旧/新版本都假设您的 API 将如何响应请求。

在 .10 版本中,通过使用 Obj-C 块可以实现相当多的可定制性。我可以拦截 RKRequest 的请求/响应,并且几乎可以覆盖 RESTKit 将要执行的任何操作。这看起来相当棒。现在有了 .20 版本,一切都变得非常……聪明(聪明的代码通常不好)。

RK 不再试图保持通用性和可塑性,而是为您做了各种“方便”的事情。这有效地剥夺了您将其塞入您的 API 所采用的任何(非理想化和现实)形状的能力。在这一点上,我已经回到使用AFNetworking编写我的所有代码。我宁愿花额外的时间编写自己的解析代码,并且知道当我的单元测试通过时,我就完成了。不要花几个小时试图让 RK 注销有意义的错误消息。

RK 的另一个真正问题是它不考虑离线。它的设计假设您的应用程序始终在线,并且只是反映了天空中的持久存储。我的应用程序都在 iOS 设备上处理内容创建,在内容创建时可能或可能不在线。这是我“定制”RK .10 的主要原因。当我看到 .20 中的变化时,我认为足够了,并回到以前的方式做事

也许我会编写自己的 CRUD/RESTful 框架,因为我开始厌倦了花时间使用非常专业的库,这些库试图做所有事情,尽可能少地留给应用程序开发人员。像这样的框架往往是瑞士军刀。他们在纸上看起来真的很棒,他们可以做任何事情,然后你实际上尝试切割一些东西并且刀片太钝或折断。好的软件很少能做得很好,如果听起来好得令人难以置信,那很可能就是这样。

于 2013-02-25T20:32:13.857 回答
3

所以你已经为每个 NSManagedObject 创建了映射(并且它们工作),但是现在你想在“Category”和“Elements”之间建立关系。

问题是没有关于每个元素所属类别的元素映射的信息(缺少唯一键)。因此,您需要将 添加cat_id到 Elements JSON 以确保此 keypath 可用。

// Elements JSON
[
  {
    "cat_id": "1313",
    "art_id": "1",
    "node": {
      "author": "name"
    },
    "small_url": 0
  },
  ...
]
// Edit keypath in relationship mapping
[elementsMapping addPropertyMapping:[RKRelationshipMapping relationshipMappingFromKeyPath:@"cat_id" toKeyPath:@"category" withMapping:categoryMapping]];

此外,您可以使用以下方法来跟踪映射是否有效。

1) -com.apple.CoreData.SQLDebug 1 // Add in "Arguments" Tab > "Arguments Passed On Launch" 
2) RKLogConfigureByName("RestKit/*", RKLogLevelTrace); // Add to AppDelegate`

编辑

映射

Elements您是否使用给定值更新了数据模型和 NSManagedObject 中的实体?好像你没有。

@interface Elements : NSManagedObject

@property (nonatomic, retain) NSNumber * catId;
@property (nonatomic, retain) NSNumber * artId;
@property (nonatomic, retain) NSString * author;
@property (nonatomic, retain) NSString * title;
@property (nonatomic, retain) NSManagedObject *category;

@end

路由

您用于获取请求的路径将不起作用,因为@"/detailaddress/:catId"它不是有效路径。一方面,您可以设置对象管理器的一般路由或定义响应描述。

 // Route for get operation
 [[RKObjectManager sharedManager].router.routeSet addRoute:[RKRoute 
      routeWithClass:[Category class] 
      pathPattern:@"/detailaddress/:catId" 
      method:RKRequestMethodGET]];

 // Response descriptor taken from the Core Data same application 
 RKResponseDescriptor *responseDescriptor = [RKResponseDescriptor responseDescriptorWithMapping:categoryMapping pathPattern:@"/detailaddress/:catId" keyPath:nil statusCodes:RKStatusCodeIndexSetForClass(RKStatusCodeClassSuccessful)];     
 [objectManager addResponseDescriptor:responseDescriptor];

 // Use general path for get request
 [[RKObjectManager sharedManager] getObjectsAtPath:@"/detailaddress" ...

但请先检查映射和数据模型是否按预期设置。

于 2013-02-19T11:15:13.003 回答