TableView的动态更新操作(无需重新加载数据源)

项目中我们经常会用到TableView展示一个Cell,Cell的数据来源于我们自定义的一个Model类,那么对于TableView我们有以下几种场景。

1. 添加操作: 在该列表页面顶部有一个按钮叫做新建,点击后进入一个新的添加页面,添加完成之后,返回到列表页更新数据。

2. 更新操作:点击列表中cell进入编辑页面,编辑页面其实就是这个Model类中属性的一个展示,对其中某些属性进行更改后,返回到列表页更新数据。

3. 删除操作:点击列表中cell进入编辑页面,该页面有个删除按钮,点击删除按钮,操作成功后,返回列表页面更新数据。

对于以上场景,一般比较笨的方法是回到列表页面后重新从服务器加载一遍数据,但太耗费资源 pass。

另一种办法就是对数据源(数组 datasource)进行更新:

  • 添加操作执行后,在数据源 datasource中insertObject:obj 然后reload列表。
  • 更新操作后,在数据源datasource找到该item进行更新然后reload列表。
  • 删除操作后,在数据源datasource找到该item进行删除操作然后reload列表。

要达到这种效果,我们需要将数据源datasource传入下一个页面或者通过消息通知机制通知datasource(当操作执行时),如果项目中该功能列表非常多的时候,每次都进行重复的操作着实有些麻烦,所以在这里,我封装了一个tableview的category用于解决以上的问题,工作原理如下:

1. 给tableview增加注册监听消息(添加,更新,删除,刷新)

- (void)addDataChangedObserver:(NSMutableArray *)datasource
                    primaryKey:(NSString *)primaryKey
                   changeBlock:(DataChangeBlock)changeBlock {

    //给类别关联数据源属性,存储tableview数据源
    objc_setAssociatedObject(self, (__bridge const void *)(kDatasource), datasource, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    //记录数据源中model的唯一标识
    objc_setAssociatedObject(self, (__bridge const void *)(kPrimaryKey), primaryKey, OBJC_ASSOCIATION_COPY_NONATOMIC);
    //记录回调的block
    objc_setAssociatedObject(self, (__bridge const void *)(kChangeBlock), changeBlock, OBJC_ASSOCIATION_COPY_NONATOMIC);

    //添加监听方法
    [self removeObserver];
    [self addObserver];

}

#pragma mark - Observer Operation

- (void)addObserver {
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(add:) name:TableViewOperationAddNotification object:nil];
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(delete:) name:TableViewOperationDeleteNotification object:nil];
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(update:) name:TableViewOperationUpdateNotification object:nil];
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(refresh:) name:TableViewOperationRefreshNotification object:nil];
}

2. 收到消息通知后调用的方法

#pragma mark - Actions

- (void)add:(NSNotification *)notification {
    NSLog(@"调用了tableview的add消息事件");
    id obj = notification.object;
    //获取数据源数据进行对象添加
    NSMutableArray *datasource = [self getPropertyKey:kDatasource];
    if (datasource && obj) {
        if (datasource.count > 0) {
            if ([obj isKindOfClass:[datasource[0] class]]) {
                [datasource insertObject:obj atIndex:0];
                [self reloadData];
            }
        }
    }

    [self callback:TableViewOperationAdd obj:obj];
}

- (void)delete:(NSNotification *)notification {
    NSLog(@"调用了tableview的delete消息事件");
    id objNotify = notification.object;

    //从数据源中删除一个对象并刷新tableview
    [self changeDataSourceWithObj:objNotify operationType:TableViewOperationDelete];
    [self callback:TableViewOperationDelete obj:objNotify];
}

- (void)update:(NSNotification *)notification {
    NSLog(@"调用了tableview的update消息事件");
    id objNotify = notification.object;

    //从数据源更新一个对象并刷新tableview
    [self changeDataSourceWithObj:objNotify operationType:TableViewOperationUpdate];
    [self callback:TableViewOperationUpdate obj:objNotify];
}

- (void)refresh:(NSNotification *)notification {
    NSLog(@"调用了tableview的refresh消息事件");
    id obj = notification.object;

    //刷新tableview
    [self reloadData];
    [self callback:TableViewOperationRefresh obj:obj];
}

- (void)callback:(TableViewOperationType)operationType obj:(id)obj {

    DataChangeBlock block = objc_getAssociatedObject(self, (__bridge const void*)kChangeBlock);
    NSIndexPath *indexPath = objc_getAssociatedObject(self, (__bridge const void*)kIndexPath);
    if (block) {
        block(operationType, indexPath, obj);
    }
}

- (void)changeDataSourceWithObj:(id)objNotify operationType:(TableViewOperationType)operationType {

    //取出数据源
    NSMutableArray *datasource = [self getPropertyKey:kDatasource];

    //取出对象主键字段名
    NSString *primaryKey = [self getPropertyKey:kPrimaryKey];

    //取出对象主键字段对应的value值
    NSString *valueNotify = [self getObjPropertyValueByKey:primaryKey obj:objNotify];
    if (objNotify) {
        [datasource enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {
            NSString *value = [self getObjPropertyValueByKey:primaryKey obj:obj];
            if ([valueNotify isEqualToString:value]) {
                if (operationType == TableViewOperationDelete) {
                    [datasource removeObject:objNotify];
                    NSLog(@"对象删除成功,刷新数据");
                } else if (operationType == TableViewOperationUpdate) {
                    [datasource replaceObjectAtIndex:idx withObject:objNotify];
                    NSLog(@"对象更新成功,刷新数据");
                }
                [self reloadData];
                *stop = YES;
            }
        }];
    }
}

主要用到了runtime的相关知识,动态的为tableview关联了属性,当程序运行到执行相应的操作时,根据关联的属性来操作数据源,达到效果。

如何使用??

1. 非常简单,在需要的viewcontroller中添加如下方法:

[self.tableview addDataChangedObserver:self.datasource primaryKey:@"pid" changeBlock:^(TableViewOperationType operationType, NSIndexPath *indexPath, id obj) {
        NSLog(@"%@", indexPath);
        NSLog(@"%ld", operationType);
        NSLog(@"%@", obj);
    }];

说明:self.datasource一个初始化了数组,存放数据源;@“pid”数据源中model对象中的标识属性,此处是Person类的中pid属性;changBlock操作时回调的block,  operationType操作类型(增、删、改、刷新),indexPath操作的行, obj操作的类,此处是person对象

2. 在需要执行操作的地方发出通知消息

- (IBAction)delete:(id)sender {
    [[NSNotificationCenter defaultCenter] postNotificationName:TableViewOperationDeleteNotification object:self.p];
}
- (IBAction)update:(id)sender {
    self.p.name = [NSString stringWithFormat:@"change_%@", self.p.name];
    [[NSNotificationCenter defaultCenter] postNotificationName:TableViewOperationUpdateNotification object:self.p];
}

大功告成,用起来是不是非常方便。

具体的源码和demo请访问

https://github.com/appleboyaug/UITableView-DataChanged

时间: 2024-08-05 10:05:08

TableView的动态更新操作(无需重新加载数据源)的相关文章

动态的计算行高 加载数据源 有多少显示多少 tableView 包含 colloctionView 显示复杂的界面写法

有时候,我们经常碰到这样的需求 先遵守代理 @interface PublishCollectionCell ()<UICollectionViewDataSource, UICollectionViewDelegate, UICollectionViewDelegateFlowLayout> 创建 _layout = [[UICollectionViewFlowLayout alloc] init]; //        layout.scrollDirection = UICollecti

tableView的懒懒的跳转方式,加载数据源方式

二者差不多,拿数据源说吧,n个section,每个section里面cell个数不固定,数据源内容不一定,导致cell形式会不一样 从数据源中取如果写if else,或者switch都不满意,也许Swift更牛一些,这里不提 说白了就是想根据所具有资源计算出不同cell对应的唯一的数据源数组的索引值 写的不好,如果有更好的希望分享一下,互相学习!(几十个scetion应该没有明显的效率诧异) 不多说上代码: -(NSArray *)arrayTitle { if (!_arrayTitle) {

WPF DataGrid双向绑定,数据源数据更改 刷新列表(无需重新加载数据源~~~)

//假设变动的属性是“IsChanged”, List<DataChangedViewModel> lstDataChanged = new List<DataChangedViewModel>(); //赋值省略 this.dataGrid.ItemsSource = lstDataChanged; private void dataGrid_SelectionChanged(object sender, SelectionChangedEventArgs e)        {

ReportViewer动态加载数据源

ReportViewer主要用于打印和导出数据到pdf或excel,接下来将简单做一张Northwind的Products表的统计报表. (最终图) 一.新建一张报表 二.添加数据集 添加xsd文件后,有两种添加数据集的方式. 第一种从工具箱中添加TableAdapter,通过sql语句连接数据库绑定数据集. 第二种添加DataTable后,手动加上需要绑定的字段,然后通过BLL层的方法返回对象绑定数据源,接下来都将采用第二种方法.          (Productid的DataType为In

Spring BeanPostProcessor与动态加载数据源配置

前言: 本文旨在介绍Spring动态配置数据源的方式,即对一个DataSource的配置诸如jdbcUrl,user,password,driverClass都通过运行时指定,而非由xml静态配置. Spring构造Context的参数一般只包含配置文件路径和类加载器,如果需要达到动态传入配置参数的目的,需要Spring在初始化数据源相关bean的时候能够对原有配置执行修改或替换,为方便处理,本文将定义一个名为DynamicDataSourceConfigHolder的公共类提供配置数据存储.

java动态编译类文件并加载到内存中

如果你想在动态编译并加载了class后,能够用hibernate的数据访问接口以面向对象的方式来操作该class类,请参考笔者的这篇博文-(该博文暂未发布) 所谓动态编译,就是在程序运行时产生java类,并编译成class文件. 一.这里介绍两种动态编译java文件的方式. 第一种:使用Runtime执行javac命令 /** * 编译java类 * 使用Runtime执行javac命令 * @param name 类的全限定包名 不带后缀 例如com.test.Notice 而不要写成com.

MaxCompute在更新插入、直接加载、全量历史表中的数据转换实践

摘要: 2018"MaxCompute开发者交流"钉钉群直播分享,由阿里云数据技术专家彬甫带来以"MaxCompute数据仓库数据转换实践"为题的演讲.本文首先介绍了MaxCompute的数据架构和流程,其次介绍了ETL算法中的三大算法,即更新插入算法.直接加载算法.全量历史表算法,再次介绍了在OLTP系统中怎样处理NULL值,最后对ETL相关知识进行了详细地介绍. 2018"MaxCompute开发者交流"钉钉群直播分享,由阿里云数据技术专家彬

02操作XML—Unity加载Xml文件方式

学习笔记,如有错误请指正.?号处也请各位指点下,谢谢. 1Unity加载本地Xml文件 xmlDocument.Load("文件绝对路径"); 2Unity加载异地Xml文件 在本地计算机上安装一个Xampp,该软件集成了Apache服务器,将一个Xml文件放到服务器下(Apache的文件放在Htdocs文件夹下) Xml内容如下: void isUserExist(string name, string pwd) { XmlElement root = readXmlFile.Get

大数据技术暑期实习七___互联网营销精准决策(加载数据源)

1. 进入Hadoop环境(在Hadoop安装目录下运行命令.若配置好ssh则可以直接运行启动命令) 2. 启动hive进程(按照网上或林子雨的配置教程来就可以,不再赘述) 进入到shell 3.加载数据到hive数据库(在项目实操中不建议查询语句为select *,而应根据列名查询,若只是查看表结构及数据效果,建议加limit,不然要机子要崩~~沙卡拉卡) hive> show tables; ##查看表 hive> desc formatted hive_table; ##描述表信息 de