CoreData:nil不是合法的NSManagedObjectContext参数

时间:2013-02-04 17:08:42

标签: ios core-data

我对Core Data非常陌生并且一直在尝试遵循许多教程,但是大多数教程都将所有Core Data方法都放到了AppDelegate中。我已经读过,最好远离AppDelegate并使用自定义数据模型类来管理这些方法。

我创建了一个自定义类来管理名为MyDataModel的所有数据。我实施了锅炉板核心数据代码。在我的一个视图控制器中,我有一个使用Core Data实现某些数据的简单方法:

- (void)getProfile {
/*
 * getProfile
 */
    NSLog(@"%@", _Model.managedObjectContext);
    Users *user = (Users *)[NSEntityDescription insertNewObjectForEntityForName:@"Users" inManagedObjectContext:_Model.managedObjectContext];

    // Set Data
    [user setValue:@"John" forKey:@"fname"];
    [user setValue:@"Smith" forKey:@"lname"];
    NSError *error;
    [_Model.managedObjectContext save:&error];
}

头文件中的代码:

#import <UIKit/UIKit.h>
#import "Users.h"
#import <CoreData/CoreData.h>
#import "DataModel.h"

@interface ProfileViewController : UIViewController

// CoreData Related
@property (strong, nonatomic) DataModel *Model;

// Instance Methods
- (void)updateProfileData;

// Core Data Method
- (void)getProfile;

@end

在视图控制器的viewDidLoad方法中调用此方法。当我运行它时,我收到以下错误:

'+entityForName: nil is not a legal NSManagedObjectContext parameter searching for entity name 'Users''

我在StackOverflow上发现了一个类似的问题可能有所帮助,但我仍然不明白解决方案究竟是什么。

'+entityForName: nil is not a legal NSManagedObjectContext parameter - Core Data

该线程声明的解决方案是他将上下文传递给ViewController。这是如何完成的?我以为我是在做那件事。

编辑:DataModel.h

#import <Foundation/Foundation.h>
#import <CoreData/CoreData.h>
#import "Workouts.h"
#import "sqlite3.h"

@interface DataModel : NSObject {
    sqlite3 *Database;
}

@property (nonatomic, strong) Workouts *currentWorkout;

// Core Data Properties
@property (nonatomic, strong) NSManagedObjectContext *managedObjectContext;
@property (nonatomic, strong) NSManagedObjectModel *managedObjectModel;
@property (nonatomic, strong) NSPersistentStoreCoordinator *storeCoordinator;

+ (DataModel *)sharedInstance;
- (void)saveContext;
- (NSURL *)applicationDocumentsDirectory;

@end

DataModel.m

#import "DataModel.h"

@implementation DataModel

#pragma mark - Core Data

+ (DataModel *)sharedInstance {
   static DataModel *sharedModel = nil;
   static dispatch_once_t onceToken;

   dispatch_once(&onceToken, ^{
       sharedModel = [[DataModel alloc] init];
       //sharedInstance.storeCoordinator = [sharedInstance storeCoordinator];
       //sharedInstance.managedObjectContext = [sharedInstance managedObjectContext];
   });
   return sharedModel;
}

- (void)saveContext {
    NSError *error = nil;
    if (_managedObjectContext != nil) {
        if ([_managedObjectContext hasChanges] && ![_managedObjectContext save:&error]) {
            NSLog(@"error: %@", error.userInfo);
        }
    }
}

#pragma mark - Core Data Stack

- (NSManagedObjectContext *)managedObjectContext {

    if (_managedObjectContext != nil) {
        return _managedObjectContext;
    }

    NSPersistentStoreCoordinator *coordinator = [self storeCoordinator];
    if (coordinator != nil) {
        _managedObjectContext = [[NSManagedObjectContext alloc] initWithConcurrencyType:NSMainQueueConcurrencyType];
        [_managedObjectContext setPersistentStoreCoordinator: coordinator];
    }
    return _managedObjectContext;
}

- (NSManagedObjectModel *)managedObjectModel {
    if (_managedObjectModel != nil) {
        return _managedObjectModel;
    }
    NSURL *modelURL = [[NSBundle mainBundle] URLForResource:@"DataModel" withExtension:@"momd"];
    _managedObjectModel = [[NSManagedObjectModel alloc] initWithContentsOfURL:modelURL];
    return _managedObjectModel;
}

- (NSPersistentStoreCoordinator *)persistentStoreCoordinator {
    if (_storeCoordinator != nil) {
        return _storeCoordinator;
    }

    NSURL *storeURL = [[self applicationDocumentsDirectory] URLByAppendingPathComponent:@"model.sqlite"];

    NSError *error = nil;
    _storeCoordinator = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:_managedObjectModel];
    if (![_storeCoordinator addPersistentStoreWithType:NSSQLiteStoreType configuration:nil URL:storeURL options:nil error:&error]) {
        NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
    }

    return _storeCoordinator;
}

#pragma mark Application's Documents Directory

- (NSURL *)applicationDocumentsDirectory {
return [[[NSFileManager defaultManager] URLsForDirectory:NSDocumentDirectory inDomains:NSUserDomainMask] lastObject];
}

@end

更新:我意识到对象是(null)的原因。我试图调用sharedInstance单例方法,但他们没有正确调用。正在调用getter并将进入一个不断调用getter的循环。我正确使用了合成变量_managedObjectContext_storeCoordinator。对象现在适当地分配内存并返回引用。感谢大家的帮助。

2 个答案:

答案 0 :(得分:4)

在Application Delegate中创建的Core Data Stack没有任何问题。你在哪里读到的情况就是这样。

调用Application Delegate来获取托管对象上下文当然被认为是糟糕的设计,但大多数人所做的是将对象委托的托管对象上下文的引用传递给使用它的其他视图控制器

答案 1 :(得分:1)

愚蠢的问题:您的模型实际上名为DataModel?