0

所以我有一个实用程序应用程序,我试图将一些文本保存到翻转视图控制器上的“收件人”和“消息:文本字段。但是,我的数据不会保存。我是目标 C 的新手,我一直是使用多个不同的教程到我完全困惑自己的地步。希望你能帮助我。不知道此时还能做什么......

FlipsideViewController.m

#import "CCCFlipsideViewController.h"
#import "CCCAppDelegate.h"
#import "CCCMainViewController.h"
#import "MessageDetails.h"

@interface CCCFlipsideViewController ()
{
   // NSManagedObjectContext *context;
}
@end

@implementation CCCFlipsideViewController
@synthesize allMessageDetails;
@synthesize managedObjectContext;

- (void)awakeFromNib
{
    [super awakeFromNib];

    CCCAppDelegate *appDelegateController = [[CCCAppDelegate alloc]init];
    self.managedObjectContext = appDelegateController.managedObjectContext;
}

- (void)viewDidLoad
{
    [super viewDidLoad];

    // Do any additional setup after loading the view, typically from a nib.

    NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
    NSEntityDescription *entity = [NSEntityDescription
                                   entityForName:@"MessageDetails" inManagedObjectContext:self.managedObjectContext];
    [fetchRequest setEntity:entity];

    NSError *error;

    self.allMessageDetails = [self.managedObjectContext executeFetchRequest:fetchRequest error:&error];

    /*
    NSManagedObject *managedObject; = [_fetchedResultsController valueForKey:@"to"];
    self.toTextField.text = managedObject to;

    messageDetails.to = [allMessageDetails firstObject];
    self.toTextField.text = messageDetails.to;

    messageDetails.message = [allMessageDetails valueForKey:@"message"];
    self.messageTextField.text = messageDetails.message;
    */
    NSLog(@"The 'to' is currently at %@ after viewdidload", self.toTextField.text);

    }

- (void)didReceiveMemoryWarning
{
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}

- (BOOL)textFieldShouldReturn:(UITextField *)textField
{
   return [textField resignFirstResponder]; //function says that if (bool) the text field is open and the keyboard hits return, text field is to resign first responder.
}

#pragma mark - Actions
- (IBAction)done:(id)sender
{
    [self.delegate flipsideViewControllerDidFinish:self];
}

- (IBAction)resignFirstResponder:(id)sender {

    [self.toTextField resignFirstResponder];
    [self.messageTextField resignFirstResponder];
    NSLog(@"Resigned First Responder");
}


- (IBAction)save:(id)sender {

    // Create a new instance of the entity managed by the fetched results controller.
   NSManagedObjectContext *context = [self.fetchedResultsController managedObjectContext];
    NSEntityDescription *entity = [[self.fetchedResultsController fetchRequest] entity];
    NSManagedObject *newManagedObject = [NSEntityDescription insertNewObjectForEntityForName:[entity name] inManagedObjectContext:context];

    // If appropriate, configure the new managed object.
    [newManagedObject setValue:self.toTextField.text forKey:@"to"];
    [newManagedObject setValue:self.messageTextField.text forKey:@"message"];

    // Save the context.
    NSError *error = nil;
    if (![context save:&error]) {
        /*
         Replace this implementation with code to handle the error appropriately.

         abort() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development. If it is not possible to recover from the error, display an alert panel that instructs the user to quit the application by pressing the Home button.
         */
        NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
        abort();
    }


}


#pragma mark -
#pragma mark Fetched results controller

- (NSFetchedResultsController *)fetchedResultsController {

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

    /*
     Set up the fetched results controller.
     */
    // Create the fetch request for the entity.
    NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] init];
    // Edit the entity name as appropriate.
    NSEntityDescription *entity = [NSEntityDescription entityForName:@"MessageDetails" inManagedObjectContext:self.managedObjectContext];
    [fetchRequest setEntity:entity];

    // Set the batch size to a suitable number.
    [fetchRequest setFetchBatchSize:20];

    // Edit the sort key as appropriate.
    NSSortDescriptor *sortDescriptor = [[NSSortDescriptor alloc] initWithKey:@"to" ascending:NO];
    NSArray *sortDescriptors = [[NSArray alloc] initWithObjects:sortDescriptor, nil];

    [fetchRequest setSortDescriptors:sortDescriptors];

    // Edit the section name key path and cache name if appropriate.
    // nil for section name key path means "no sections".
    NSFetchedResultsController *aFetchedResultsController = [[NSFetchedResultsController alloc] initWithFetchRequest:fetchRequest managedObjectContext:self.managedObjectContext sectionNameKeyPath:nil cacheName:@"Root"];
    aFetchedResultsController.delegate = self;
    self.fetchedResultsController = aFetchedResultsController;


    NSError *error = nil;
    if (![_fetchedResultsController performFetch:&error]) {
        /*
         Replace this implementation with code to handle the error appropriately.

         abort() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development. If it is not possible to recover from the error, display an alert panel that instructs the user to quit the application by pressing the Home button.
         */
        NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
        abort();
    }

    return _fetchedResultsController;
}


@end
4

3 回答 3

1

我没有查看您的所有代码,因为顶部附近存在一个问题,它否定了您此后所做的一切。不要在 awakeFromNib 或其他任何地方分配/初始化您的应用程序委托。您的应用程序委托的唯一实例已经存在(我不知道当有多个应用程序委托时会发生什么)。

CCCFlipsideViewController 需要通过另一种方式获得对托管对象上下文的访问权限。也许 CCCMainViewController(或另一个视图控制器)可以设置 CCCFlipsideViewController 的 managedObjectContext 属性。如果 CCCMainViewController 无权访问托管对象上下文,则让应用委托将该上下文传递给它。

示例:应用委托在根视图控制器上设置 managedObjectContext 属性;反过来,根视图控制器在子视图控制器(如您的反面 VC)上设置 managedObjectContext 属性等。

于 2013-09-28T20:24:41.717 回答
0

老实说,我认为你在这段代码上做了很多工作...... ;)

好的,首先,在您的保存方法中,不要创建另一个 NSManagedObjectContext,使用您已经声明的实例变量“managedObjectContext”。

其次,我认为你把事情弄得太复杂了......一旦你创建了 NSManagedObject 子类并在 App Delegate 中设置了所有内容,存储核心数据实际上非常简单......

似乎您在代码中此时不需要来自“fetchedResultsController”的任何信息,因为您正在保存,而不是获取。也许尝试将您的保存方法更改为:

- (IBAction)save:(id)sender {

    NSEntityDescription *entity = [NSEntityDescription insertNewObjectForEntityForName:@"MessageDetails" inManagedObjectContext:self.managedObjectContext];

    // If appropriate, configure the new managed object.
    [entity setValue:self.toTextField.text forKey:@"to"];
    [entity setValue:self.messageTextField.text forKey:@"message"];

    // Save the context.
    NSError *error = nil;
    [self.managedObjectContext save:&error]

    if (error) {
        /*
         Replace this implementation with code to handle the error appropriately.

         abort() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development. If it is not possible to recover from the error, display an alert panel that instructs the user to quit the application by pressing the Home button.
     */
        NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
        abort();
    }

}


编辑:并从应用程序委托中获取托管对象上下文...

在您的@synthesize 之后,为 App Delegate 创建一个变量。

AppDelegate* appDelegateController;

并在 viewDidLoad 中对其进行初始化:

appDelegateController = (AppDelegate*)[[UIApplication sharedApplication] delegate];

在 viewDidLoad 之后(或任何你想要的地方),你可以坚持一个方法来声明托管对象上下文:

- (NSManagedObjectContext*)managedObjectContext {
    return appDelegateController.managedObjectContext;
}

然后返回 viewDidLoad,调用该方法:

self.managedObjectContext = [self managedObjectContext];
于 2013-09-28T20:40:47.753 回答
0

您似乎从未真正将 self.messageTextField.text 或 self.toTextField.text 设置为任何内容——您已经在设置这些字段的 viewDidLoad 方法中注释掉了代码。bilobatum 关于您的 AppDelegate 问题也是完全正确的——您也可以使用类似的东西

[((NSObject*)[UIApplication sharedApplication].delegate) valueForKey: @"managedObjectContext"];

如果您想快速解决这个问题,请为您的应用程序获取应用程序委托,尽管长期 bilobatum 的解决方案是更好的设计。

于 2013-09-28T20:32:54.777 回答