如何从 UITableView 中删除一行

Posted

技术标签:

【中文标题】如何从 UITableView 中删除一行【英文标题】:How to delete a row from UITableView 【发布时间】:2010-12-21 09:45:59 【问题描述】:

我正在尝试从我的表格视图中删除一行,但到目前为止一直没有成功。我不断收到此错误:

“由于未捕获的异常而终止应用程序 'NSInternalInconsistencyException',原因:'无效更新:无效 第 0 节中的行数。包含在第 0 节中的行数 更新后的现有节(5)必须等于 更新前该部分中包含的行 (5),加号或减号 从该部分插入或删除的行数(0 插入, 1 已删除)。'” 我用来填充表格视图的数组是 在那个类中声明,我也从一个 sqlite 数据库。

我用来尝试删除行的代码如下。

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

if (editingStyle == UITableViewCellEditingStyleDelete) 
    // Delete the row from the data source

    [categoryArray objectAtIndex:indexPath.row];
    [tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:YES];

   
else if (editingStyle == UITableViewCellEditingStyleInsert) 
    // Create a new instance of the appropriate class, insert it into the array, and add a new row to the table view
   

现在是我希望得到回答的问题:

-如何正确删除行?

-如果 tableView 中只剩下一行,我会面临删除该行吗?

-我可以将方法中使用的 tableView 更改为我在其中声明的 UITableView 我的 .h 文件?

非常感谢。


编辑

-完整的.m代码

#import "DeleteCategoryTableView.h"
#import "KeyCryptAppAppDelegate.h"


@implementation DeleteCategoryTableView
@synthesize categoryArray;

#pragma mark -
#pragma mark Initialization

-(void) initializeCategoryArray 

sqlite3 *db= [KeyCryptAppAppDelegate getNewDBConnection];
KeyCryptAppAppDelegate *appDelegate = (KeyCryptAppAppDelegate *)[[UIApplication sharedApplication] delegate];

const char *sql = [[NSString stringWithFormat:(@"Select Category from Categories;")]cString];


sqlite3_stmt *compiledStatement;



if (sqlite3_prepare_v2(db, sql, -1, &compiledStatement, NULL)==SQLITE_OK)

    while(sqlite3_step(compiledStatement) == SQLITE_ROW)
        [categoryArray addObject:[NSString stringWithUTF8String:(char*) sqlite3_column_text(compiledStatement, 0)]];


else 
    NSAssert1(0,@"Error preparing statement", sqlite3_errmsg(db));

sqlite3_finalize(compiledStatement);

/*
- (id)initWithStyle:(UITableViewStyle)style 
// Override initWithStyle: if you create the controller programmatically and want to perform customization that is not appropriate for viewDidLoad.
if ((self = [super initWithStyle:style])) 

return self;

*/
#pragma mark -
#pragma mark View lifecycle
- (void)viewDidLoad 
self.title = NSLocalizedString(@"Delete Categories",@"Delete your Categories");
categoryArray = [[NSMutableArray alloc]init];
[self initializeCategoryArray];
[super viewDidLoad];

// Uncomment the following line to display an Edit button in the navigation bar for this view controller.
self.navigationItem.rightBarButtonItem = self.editButtonItem;

/*
- (void)viewWillAppear:(BOOL)animated 
[super viewWillAppear:animated];

*/
/*
- (void)viewDidAppear:(BOOL)animated 
[super viewDidAppear:animated];

*/
/*
- (void)viewWillDisappear:(BOOL)animated 
[super viewWillDisappear:animated];

*/
/*
- (void)viewDidDisappear:(BOOL)animated 
[super viewDidDisappear:animated];

*/
/*
// Override to allow orientations other than the default portrait orientation.
- (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation 
// Return YES for supported orientations
return (interfaceOrientation == UIInterfaceOrientationPortrait);

*/


#pragma mark -
#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.categoryArray count];



// Customize the appearance of table view cells.
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath 

static NSString *CellIdentifier = @"Cell";

UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
if (cell == nil) 
    cell = [[[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier] autorelease];


// Configure the cell...
NSUInteger row = [indexPath row];
cell.text = [categoryArray objectAtIndex:row];
return cell;

/*
// Override to support conditional editing of the table view.
- (BOOL)tableView:(UITableView *)tableView canEditRowAtIndexPath:(NSIndexPath *)indexPath 
// Return NO if you do not want the specified item to be editable.
return YES;

*/
// Override to support editing the table view.
- (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath 

if (editingStyle == UITableViewCellEditingStyleDelete) 
    // Delete the row from the data source

    [categoryArray removeObjectAtIndex:indexPath.row];
    [tableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:YES];
    [deleteCategoryTable reloadData];

    NSString *selectedCategory = [categoryArray objectAtIndex:indexPath.row];

    sqlite3 *db= [KeyCryptAppAppDelegate getNewDBConnection];
    KeyCryptAppAppDelegate *appDelegate = (KeyCryptAppAppDelegate *)[[UIApplication sharedApplication] delegate];
    const char *sql = [[NSString stringWithFormat:@"Delete from Categories where Category = '%@';", selectedCategory]cString];

    sqlite3_stmt *compiledStatement;


    if (sqlite3_prepare_v2(db, sql, -1, &compiledStatement, NULL)==SQLITE_OK)
    
        sqlite3_exec(db,sql,NULL,NULL,NULL);

    
    else 
        NSAssert1(0,@"Error preparing statement", sqlite3_errmsg(db));
    
    sqlite3_finalize(compiledStatement);




else if (editingStyle == UITableViewCellEditingStyleInsert) 
    // Create a new instance of the appropriate class, insert it into the array, and add a new row to the table view
   

/*
// Override to support rearranging the table view.
- (void)tableView:(UITableView *)tableView moveRowAtIndexPath:(NSIndexPath *)fromIndexPath toIndexPath:(NSIndexPath *)toIndexPath 

*/
/*
// Override to support conditional rearranging of the table view.
- (BOOL)tableView:(UITableView *)tableView canMoveRowAtIndexPath:(NSIndexPath *)indexPath 
// Return NO if you do not want the item to be re-orderable.
return YES;

*/
#pragma mark -
#pragma mark Table view delegate
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath 
// Navigation logic may go here. Create and push another view controller.
/*
 <#DetailViewController#> *detailViewController = [[<#DetailViewController#> alloc] initWithNibName:@"<#Nib name#>" bundle:nil];
 // ...
 // Pass the selected object to the new view controller.
 [self.navigationController pushViewController:detailViewController animated:YES];
 [detailViewController release];
 */

#pragma mark -
#pragma mark Memory management

- (void)didReceiveMemoryWarning 
// Releases the view if it doesn't have a superview.
[super didReceiveMemoryWarning];

// Relinquish ownership any cached data, images, etc that aren't in use.


- (void)viewDidUnload 
// Relinquish ownership of anything that can be recreated in viewDidLoad or on demand.
// For example: self.myOutlet = nil;



- (void)dealloc 
[super dealloc];



@end

【问题讨论】:

【参考方案1】:

1) 删除的问题是

[categoryArray removeObjectAtIndex:indexPath.row];

也将其从数据库中删除。

删除行后,使用[tableView reloadData]重新加载tableView

2) 如果只有一项也没问题

编辑:

这就是问题

[categoryArray removeObjectAtIndex:indexPath.row];
NSString *selectedCategory = [categoryArray objectAtIndex:indexPath.row];

在访问同一 indexPath.row 的值之后,您已经从数组中删除了 indexPath.row 的值,显然它只会删除下一个值.. :)

那就这么说吧

NSString *selectedCategory = [categoryArray objectAtIndex:indexPath.row]

[categoryArray removeObjectAtIndex:indexPath.row];之前的行

【讨论】:

嘿伙计,谢谢你的回答我不敢相信我忘记了这么简单的事情。再次感谢! 嘿,关于您的解决方案,我刚刚意识到我遇到了一个问题。我无法删除最后一行,它崩溃了,这有点傻。我在删除唯一一行时也遇到了问题。有什么想法吗? *** 由于未捕获的异常 'NSRangeException' 导致应用程序终止,原因:'*** -[NSMutableArray objectAtIndex:]: index 4 beyond bounds [0 .. 3]' 还有如果 i;m剩下 2 行,我删除第一行(而不是最后一行),最后一行被删除。 您是否在表格中手动添加任何行?我要问的是,tableView 中是否有任何默认行? 默认行?我不确定你的意思。但是我的表视图中的所有行都是从一个可变数组中获取的,该数组从一个 sqlite 数据库中获取它的对象,到目前为止我还没有以编程方式添加任何行。【参考方案2】:

这是您需要做的所有。归根结底,可以很好地实现一种方法。

斯威夫特 2

func tableView(tableView: UITableView, commitEditingStyle editingStyle: UITableViewCellEditingStyle, forRowAtIndexPath indexPath: NSIndexPath) 
    if editingStyle == .Delete 

        //1. remove data from model
        data.removeAtIndex(indexPath.row) 

        //2. remove row from view
        tableView.deleteRowsAtIndexPaths([indexPath], withRowAnimation: .Fade) 

        //3. custom method to update your view after removing
        updateView() 
    

斯威夫特 3

func tableView(_ tableView: UITableView, commit editingStyle: UITableViewCellEditingStyle, forRowAt indexPath: IndexPath) 
    if editingStyle == .delete 

        //1. remove data from model
        data.remove(at: indexPath.row)

        //2. remove row from view
        tableView.deleteRows(at: [indexPath as IndexPath], with: .fade)

        //3. custom method to update your view after removing
        updateView()
    

【讨论】:

【参考方案3】:

您的代码很好,希望 withRowAnimation: 不希望是 BOOL 而是 UITableViewRowAnimation 成员。例如。 UITableViewRowAnimationRight 让它滑到桌子的右侧。

另外:你只是从数组中获取对象,你必须删除它!如果它是一个可变数组,你可以使用[categoryArray removeObjectAtIndex:indexpath.row]

【讨论】:

推测您可以使用 addRowsAtIndexPaths: withRowAnimation: 之类的东西来添加,或者我弄错了 是的,还有对应功能。只需确保将数据添加到您的数组(或您使用的任何东西)中,并且您能够在调用函数之前返回正确的新行数 如果我的 indexPath 为 2,这是否意味着 insertRowsAtIndexPath:[NSIndexPath initWithNum:2](或类似的东西)会在第二个元素之前或之后插入它们? 它会将其添加为第二个元素。所以之前的第 2 个元素将成为第 3 个,之前的第 3 个元素将成为第 4 个,依此类推。但是,它只会为该行设置动画并重新加载它。您需要确保该元素也位于数组中的相同位置,否则结果可能看起来有些有趣/错误【参考方案4】:

为此,不要尝试删除行尝试从数组或数据源中删除(如果数组正在从数据库准备)然后尝试通过简单地调用数据库函数来重新加载数组。

然后

[table reloadData].

这是正确的做法。

【讨论】:

【参考方案5】:

从控制台读取问题所在。

你有一个方法numberOfRowsInSection...假设它设置为return 5;

当您从表视图中删除一行时,您还需要更新此方法...因此它会从该方法中寻找您到 return 4;

基本上,您需要确保表格视图中的行数比删除行之前少一个。

【讨论】:

以上是关于如何从 UITableView 中删除一行的主要内容,如果未能解决你的问题,请参考以下文章

Swift 3.0 无法从 UITableView 中删除一行

从 UItableview 删除一行时崩溃

使用 UIButton 从其他 UIViewController 中删除一行

从 UITableView 中删除最后一行 - 因 NSInternalInconsistencyError 而崩溃

在 Swift 中从 UITableView 中删除一行?

删除 UITableView 部分的最后一行而不删除部分?