iOS Почему мое приложение не загружает данные в табличное представление, пока оно не будет перезапущено?

В своем стремлении узнать, как правильно загружать данные в сгруппированное табличное представление из Core Data, я создал очень простое маленькое приложение с двумя представлениями. Я использую MagicalRecord в качестве интерфейса с CD.

Текущая проблема заключается в том, что, хотя входные данные сохраняются в моем магазине, как и ожидалось, табличное представление не обновляется при сохранении для отображения новых данных. Однако, если я остановлю приложение в симуляторе, а затем перезапущу его из Xcode, новые данные отобразятся в табличном представлении.

Я использую NotificationCenter, а не делегирование для связи. Вот код из моего начального контроллера представления (содержащего и делегирующего табличное представление), за которым следует код из AddViewController, используемый для добавления данных. Может ли кто-нибудь с опытом работы с MagicalRecord пролить на меня свет? Спасибо!

//
//  ViewController.m
//  MRGroupingTest
//
//  Created by Tim Jones on 1/9/14.
//  Copyright (c) 2014 TDJ. All rights reserved.
//

#import "ViewController.h"
#import "ListActivity.h"

@interface ViewController ()

{
    NSFetchedResultsController *frc;
}


@end

@implementation ViewController

- (void)viewDidLoad
{
    [super viewDidLoad];
    // Do any additional setup after loading the view, typically from a nib.

    [self refreshData];
    frc = [ListActivity MR_fetchAllGroupedBy:@"category" withPredicate:nil sortedBy:@"name" ascending:YES];
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(notificationNewActivityAdded:) name:@"newActivityAdded" object:nil];

}

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

-(void) notificationNewActivityAdded:(NSNotification*)notification
{
    [self.myTableView reloadData];
}



#pragma mark Table View data source

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
{
    return [[frc sections] count];
}

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
    id<NSFetchedResultsSectionInfo> sectionInfo = [[frc sections] objectAtIndex:section];
    int rows = [sectionInfo numberOfObjects];
    return rows;

}

- (NSString *)tableView:(UITableView *)tableView titleForHeaderInSection:(NSInteger)section
{
    id <NSFetchedResultsSectionInfo> sectionInfo = [[frc sections] objectAtIndex:section];
    return [sectionInfo name];

}

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    static NSString *CellIdentifier = @"Cell";
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier forIndexPath:indexPath];

    // Configure the cell...
    if ([frc sections] > 0)
    {
        ListActivity *activity = [frc objectAtIndexPath:indexPath];
        cell.textLabel.text = activity.name;

    }

    return cell;

}

-(void) refreshData
{

    frc = [ListActivity MR_fetchAllGroupedBy:@"category" withPredicate:nil sortedBy:@"name" ascending:YES];

    [self.myTableView reloadData];
}


#pragma mark Table View delegate

@end

А вот код из AddViewController:

//
//  AddViewController.m
//  MRGroupingTest
//
//  Created by Tim Jones on 1/9/14.
//  Copyright (c) 2014 TDJ. All rights reserved.
//

#import "AddViewController.h"

@interface AddViewController ()

@end

@implementation AddViewController

- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
{
    self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
    if (self) {
        // Custom initialization
    }
    return self;
}

- (void)viewDidLoad
{
    [super viewDidLoad];
    // Do any additional setup after loading the view.
}

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

- (IBAction)saveAction:(UIBarButtonItem *)sender
{
    NSManagedObjectContext *localContext = [NSManagedObjectContext MR_contextForCurrentThread];

    ListActivity * thisActivity = [ListActivity MR_createInContext:localContext];
    thisActivity.name =  self.activityField.text;
    thisActivity.category = self.categoryField.text;
    [localContext MR_saveToPersistentStoreAndWait];
    //Inform app
    [[NSNotificationCenter defaultCenter] postNotificationName:@"newActivityAdded" object:localContext];
    //dismiss view
    [self.navigationController dismissViewControllerAnimated:YES completion:nil];


}

- (IBAction)cancelAction:(UIBarButtonItem *)sender
{
    NSManagedObjectContext *localContext = [NSManagedObjectContext MR_contextForCurrentThread];
    [[NSNotificationCenter defaultCenter] postNotificationName:@"newActivityAdded" object:localContext];
    //dismiss view
    [self.navigationController dismissViewControllerAnimated:YES completion:nil];

}


@end

person rattletrap99    schedule 10.01.2014    source источник


Ответы (1)


Вам нужно перезагрузить данные TableView, т.е. вызвать функцию reloadData

person Lefteris    schedule 10.01.2014
comment
Спасибо за ответ! Я явно упускаю здесь что-то критичное. Где я должен добавить этот вызов, в частности? - person rattletrap99; 11.01.2014
comment
Ну, у него есть вызов reloadData, в refreshData. Но есть вероятность, что где-то есть нулевой указатель или фальшивая вторая копия объекта, поэтому reloadData не вызывается в реальном TableView. - person Hot Licks; 11.01.2014
comment
@HotLicks - ваш комментарий вызвал опасения, что может возникнуть конфликт между локальной переменной tableView в стандартном исходном коде данных, который я использую, и свойством tableView в моем ViewController. Поэтому я вернулся и изменил его вместе с еще парой изменений, отражающих предложение первого комментатора. См. редактирование для пересмотренного кода ViewController. Все еще не работает. - person rattletrap99; 11.01.2014
comment
@TimJones - Похоже, ты связал себя узлом. Вам нужно вернуться назад и тщательно все обдумать, а также выполнить некоторую отладку (проверить, выполняете ли вы когда-либо reloadData и хорошо ли работает указатель на TableView, когда вы это делаете). - person Hot Licks; 11.01.2014
comment
Я был бы не в первый раз в узле. :) Я сделаю, как вы предлагаете, и отчитаюсь. Спасибо! - person rattletrap99; 11.01.2014
comment
@HotLicks -- Хорошо, более знающий друг помог мне разобраться с кодом сегодня утром, и хотя он тоже обнаружил, что поведение сбивает с толку, мы, наконец, обнаружили пару избыточностей и, что настораживает, то, что строка addObserver в viewDidLoad была закомментирована, даже хотя этого нет в коде, который я изначально опубликовал. Не могу этого объяснить, хотя я явно единственный подозреваемый. В любом случае, теперь это работает, поэтому я собираюсь отметить этот вопрос решенным. Некоторые другие вопросы о методах сортировки и группировки MagicalRecord остаются, но я задам их отдельно. Большое спасибо вам, ребята! - person rattletrap99; 12.01.2014