使用 Core Data 的 View Controller 中的表视图

Posted

技术标签:

【中文标题】使用 Core Data 的 View Controller 中的表视图【英文标题】:Table View in View Controller using Core Data 【发布时间】:2013-06-20 13:34:28 【问题描述】:

我正在使用核心数据并尝试将表格视图单元格添加到视图控制器中的表格视图中。当我在表视图控制器中实现它时,应用程序就会运行。当我在表视图是另一个视图控制器的一部分时运行它时,应用程序不会运行。它在代码中显示错误:在 DeviceViewController 类型的对象上找不到属性表视图。 DeviceViewController.h 有一个表视图数据源和委托。我的代码是:

 #import "DeviceViewController.h"
#import "DeviceDetailViewController.h"

 @interface DeviceViewController ()
@property (strong) NSMutableArray *devices;
@end

@implementation DeviceViewController

- (NSManagedObjectContext *)managedObjectContext 
NSManagedObjectContext *context = nil;
id delegate = [[UIApplication sharedApplication] delegate];
if ([delegate performSelector:@selector(managedObjectContext)]) 
context = [delegate managedObjectContext];

return context;


- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil

self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
if (self) 
// Custom initialization

return self;

- (void)viewDidLoad

[super viewDidLoad];

// self.navigationItem.rightBarButtonItem = self.editButtonItem;
     

- (void)viewDidAppear:(BOOL)animated

[super viewDidAppear:animated];

 // Fetch the devices from persistent data store
NSManagedObjectContext *managedObjectContext = [self managedObjectContext];
NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] initWithEntityName:@"Device"];
self.devices = [[managedObjectContext executeFetchRequest:fetchRequest error:nil]             mutableCopy];

[self.tableView reloadData];


- (void)didReceiveMemoryWarning

[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.


#pragma mark - Table view data source

- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView

// Return the number of sections.
return 1;


 - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section

// Return the number of rows in the section.
return self.devices.count;


- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath     *)indexPath

static NSString *CellIdentifier = @"Cell";
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier           forIndexPath:indexPath];

// Configure the cell...
NSManagedObject *device = [self.devices objectAtIndex:indexPath.row];
[cell.textLabel setText:[NSString stringWithFormat:@"%@ ", [device valueForKey:@"name"]]];

return cell;



- (BOOL)tableView:(UITableView *)tableView canEditRowAtIndexPath:(NSIndexPath *)indexPath

// Return NO if you do not want the specified item to be editable.
return YES;



  - (void)tableView:(UITableView *)tableView commitEditingStyle:        (UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath

NSManagedObjectContext *context = [self managedObjectContext];

if (editingStyle == UITableViewCellEditingStyleDelete) 
// Delete object from database
[context deleteObject:[self.devices objectAtIndex:indexPath.row]];

NSError *error = nil;
   if (![context save:&error]) 
NSLog(@"Can't Delete! %@ %@", error, [error localizedDescription]);
return;


// Remove device from table view
[self.devices removeObjectAtIndex:indexPath.row];
[self. otableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath]          withRowAnimation:UITableViewRowAnimationFade];
  

【问题讨论】:

【参考方案1】:

在我看来,在这种情况下,它与 CoreData 无关。

您应该将UITableViewDataSourceUITableViewDelegate 协议添加到您的VC。 之后,您应该创建一个 UITableView(您可以在界面生成器中完成)并将数据源和委托连接到您的视图控制器。

您还可以在 VC 中为 UITableView 添加属性。

当你使用UITableViewController 时,所有这些都是从 Xcode 完成的,但如果你在常规 UIViewController 中使用 UITableView,你应该注意这些东西。

祝你好运!

编辑:

将此添加为属性

@property (nonatomic, strong) UITableView *deviceTableView;

viewDidLoad方法改成这样:

- (void)viewDidLoad 
    [super viewDidLoad];
    _deviceTableView = [[UITableView alloc] initWithFrame:self.view.frame];
    _deviceTableView.delegate = self;
    _deviceTableView.dataSource = self;
    [self.view addSubview:_deviceTableView];
    [_deviceTableView reloadData];
 

删除您在viewDidAppear 中写入的内容,并将此文件中的所有self.tableView 替换为_deviceTableView

【讨论】:

我这样做了,但它在加载时崩溃了。显示 SIGBART 错误 您收到的异常消息是什么? 我更改了一些内容并将其添加到我的 viewdidappear 函数中: DeviceViewController *deviceViewController = [[DeviceViewController alloc]init]; deviceViewController.tableView = self.tableView; self.tableView.dataSource = deviceViewController; self.tableView.delegate = deviceViewController;现在,它在 [self.tableView reloadData] 上显示断点; 你在viewDidAppear写的东西是错的。删除它们。请参阅我编辑的帖子。还可以阅读一些关于 ViewControllers 和 UITableView 的教程。【参考方案2】:

另一个可能导致崩溃的问题是,当您尝试执行获取请求时,如果您的 NSManagedObjectContext 为 nil。我会添加一个if([self managedObjectContext]) 在您的任何视图层次结构方法中使用它之前。

此外,在使用从故事板实例化的视图时,您应该小心。 Nikola 的回答是在 viewDidLoad 中添加配置视图控制器的代码,只要您没有在界面构建器中将 UITableView 拖到 UIViewController 实例上,就可以了。如果您已经在界面生成器中创建并链接了一个表格视图,您将不必要地用一个新的替换它,所以在 _deviceTableView = [[UITableView alloc] initWithFrame:self.view.frame]; 行,你应该添加一个if(!_deviceTableView) 声明,因为它会阻止您向视图添加第二个表视图。问题是如果您已经在 IB 中添加了一个 tableview,即使您更改了 tableview 属性,它也会被您的视图保留,因此可能会显示两个表。

或者,您可以在表视图属性的 getter 方法中使用“延迟实例化”,以确保在您访问它时对其进行初始化。

最后,为了帮助我们更好地了解您的应用崩溃的原因,从 Xcode 的控制台获取异常日志会很有帮助。

【讨论】:

我认为问题是当 NSManagedObjectContext 正在执行 fetch 请求时,你能告诉我要写什么代码吗? 我知道viewdidappear函数有问题,只是不知道怎么改,能不能帮帮我 我在 viewDidAppear 函数中注意到的另外几件事,它们都不能与崩溃直接相关,但它们都是您可能想要修复的问题。首先,在您的 managedObjectContext getter 中,当您可能需要 respondsToSelector: 时调用 if ([delegate performSelector:@selector(managedObjectContext)]),因为如果 performSelector 不存在,它仍然会使您的应用程序崩溃。第二件事是,当对 tableview 使用 fetch 请求时,您必须使用排序描述符,以便每次执行时对象都以相同的顺序返回。 虽然在我们看到崩溃日志之前,不幸的是我认为我们无法确定崩溃的起源。但是,我的直觉告诉我,您的自定义视图控制器上没有属性 tableview。这就是为什么它说property tableview not found。如果您的类是 UITableViewController 的子类,则属性为 @property(nonatomic, retain) UITableView *tableView(带有大写“V”) 我添加了自定义表格视图控制器。即使这样,这个问题也正在发生。我知道视图确实出现方法存在问题,因为当我删除它时,一切正常,除了我在另一个 VC 上的保存方法。

以上是关于使用 Core Data 的 View Controller 中的表视图的主要内容,如果未能解决你的问题,请参考以下文章

如何将展开和折叠列表行与 Core Data 元素一起使用?

如何在 Core Data 中正确保存?

Core Data - 数据传输 SwiftUI

SwiftUI Core Data 删除实体的相关数据

如何在类方法中执行 Core Data NSFetchRequest?

为啥 Core Data 上下文对象必须通过环境变量传递?