1.友盟错误信息

Invalid update: invalid number of rows in section .  The number of rows contained in an existing section after the update ()
must be equal to the number of rows contained in that section before the update (), plus or minus the number of rows inserted or deleted
from that section ( inserted, deleted) and plus or minus the number of rows moved into or out of that section ( moved in, moved out).
(null)

2.错误信息解析

上面的错误信息,大意是:调用insertRowsAtIndexPaths或deleteRowsAtIndexPaths时,表格 的 行数 一定要与数据源的数量一致。

2.1原因分析1:异步线程更新数据源

下面使用一个Demo来复现这种情况:

- (void)p_addRow {
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, ), ^{
[NSThread sleepForTimeInterval:0.5]; //模拟网络数据加载
[self.arrData addObject:@"cell number 1 --"]; dispatch_async(dispatch_get_main_queue(), ^{
[self.tableView beginUpdates];
[self.tableView insertRowsAtIndexPaths:@[[NSIndexPath indexPathForRow:self.arrData.count - inSection:]] withRowAnimation:UITableViewRowAnimationFade];
[self.tableView endUpdates];
});
});
} - (void)p_deleteRow {
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, ), ^{
[NSThread sleepForTimeInterval:0.5]; //模拟网络数据加载
[self.arrData removeObjectAtIndex:]; dispatch_async(dispatch_get_main_queue(), ^{
[self.tableView beginUpdates];
[self.tableView deleteRowsAtIndexPaths:@[[NSIndexPath indexPathForRow:self.arrData.count - inSection:]] withRowAnimation:UITableViewRowAnimationFade];
[self.tableView endUpdates];
});
});
}

这两个方法调用的时候,我们先按照常规的操作调用,每点击一次,添加/删除一条数据,这种情况下,从实际效果可以看到不会出现崩溃的现象。

现在用一个定时器来模拟手点的效果,经过试验得知,当定时器执行的时间间隔过快,而网络数据反应太慢的情况,会出现崩溃的现象,代码如下:

- (void)leftButton {
self.timer = [NSTimer scheduledTimerWithTimeInterval:0.01 target:self selector:@selector(p_deleteRow) userInfo:nil repeats:YES];
} - (void)rightButton {
[NSTimer scheduledTimerWithTimeInterval:0.01 target:self selector:@selector(p_addRow) userInfo:nil repeats:NO];
} - (void)p_addRow {
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, ), ^{
[NSThread sleepForTimeInterval:0.5]; //模拟网络数据加载
[self.arrData addObject:@"cell number 1 --"]; dispatch_async(dispatch_get_main_queue(), ^{
[self.tableView beginUpdates];
[self.tableView insertRowsAtIndexPaths:@[[NSIndexPath indexPathForRow:self.arrData.count - inSection:]] withRowAnimation:UITableViewRowAnimationFade];
[self.tableView endUpdates];
});
});
} - (void)p_deleteRow {
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, ), ^{
[NSThread sleepForTimeInterval:0.5]; //模拟网络数据加载
[self.arrData removeObjectAtIndex:]; dispatch_async(dispatch_get_main_queue(), ^{
[self.tableView beginUpdates];
[self.tableView deleteRowsAtIndexPaths:@[[NSIndexPath indexPathForRow:self.arrData.count - inSection:]] withRowAnimation:UITableViewRowAnimationFade];
[self.tableView endUpdates];
});
});
}

崩溃日志如下:

这样看还是有点抽象,我们加一个变量来保存当前表格的行数以及在执行insertRowsAtIndexPaths时数据源的表格行数:

从结果可以看到,由于获取数据是在异步线程里面进行的,导致在执行insertRowsAtIndexPaths时,实际上数据源已经进行了多次插入操作,这样,只执行一次insertRowsAtIndexPaths,便会出现崩溃现象了。

2.2原因分析2:主线程更新数据源

上面小节是在异步线程中更新数据源,这里我们把数据源的放在主线程,看看实际效果:

使用上面的代码,我们可以看到没有出现崩溃的问题,那么是不是表示在主线程更新数据源就没有问题呢?我们试一下这种情况:如果通过网络加载或其他方式加载数据的时候,数据源有多条数据呢?还是看代码演示:

从这里可以看到,如果数据源更新了多条数据,仍然在insertRowsAtIndexPaths方法里面只加了一条数据时,便会出现崩溃现象。

2.3其他情况

上两节的内容分析的崩溃原因,有一个共同点:下面这两个数字是不一样的(删除的话,前面那个数字应该小于后面那个数字,如果不是这个规则,应该是网络又更新了数据源)。

The number of rows contained in an existing section after the update () must be equal to the number of rows contained in that section before the update ()

但从UMeng的崩溃日志看,实际上还有一种情况,那就是这两个数字是一样的。这种情况是怎么发生的呢?我们修改一下代码:

从代码可以看到:更新数据源之后,先调用了一下[self.tableView reloadData],这个时候因为表格数据源已经更新,并且表格也已经更新,这个时候再调用一次insertRowsAtIndexPaths方法,就会接着导致数据源和表格行不一致,从而导致崩溃。

3.解决方案

通过对我们工程代码的分析,数据源的操作是在主线程(对涉及到数据源的地方,都打印了线程的日志信息,显示当前线程为主线程),因此初步判断出现崩溃的原因为2.2小节和2.3小节所描述的。这两种情况,我们先用Demo来演示一下解决方案看看是不是有效的:

逐步放开两个注释内容,可以看到,这三个条件下,都不会崩溃。

同样的,演示一下删除:

逐步放开两个注释内容,可以看到,这三个条件下,也不会崩溃。

一个友盟BUG的思考和分析:Invalid update的更多相关文章

  1. android第三方分享之友盟社会化组件

    前言 现在几乎所有的app都带有分享功能,第一为了更好地推广自己的产品,第二作为使用者也能及时的把自己觉得好的文章,话题,app分享到社交平台供大家一起学习和使用.开发中虽然android系统自带分享 ...

  2. iOS开发-友盟分享(1)

    1.集成友盟分享,需要先注册一个友盟账号,注册地址 友盟开发者平台官网  友盟集成文档 友盟sdk下载地址友盟sdk下载地址 2,成功下载sdk集成后,微信分享需要配置一下 新浪微博 之类到同样配置就 ...

  3. 友盟分享--集成QQ和微信

    随着社交工具的应用范围越来越广,分享一些内容的功能也开始要求实现了. 用得比较多的第三方,比如说友盟,比如说Share等等... 前几天刚用友盟写了集成QQ和微信客户端的功能,觉得有必要分享一下. 在 ...

  4. AppDelegate减负之常用三方封装 - 友盟推送篇

    之前分享过集成友盟推送的方法, 需要的朋友可以查看一下链接: http://www.cnblogs.com/zhouxihi/p/6533058.html 一般开发中我们比较多使用的三方有友盟推送, ...

  5. 线上应用bug跟踪查找-友盟统计

    线上的应用只要用心点点都能发现些bug,连微信,QQ也不列外.但是bug中最严重的算是闪退了,这导致了用户直接不能使用我们的app. 我们公司是特别注重用户反馈和体验的,我们会定期打电话咨询用户的使用 ...

  6. 如何通过友盟分析发布后App崩溃日志-b

    要分析崩溃日志,首先需要保留发布时的编译出来的.xcarchive文件.这个文件包含了.DSYM文件. 我一般的做法是,发布成功后,把这个文件.xcarchive直接提交到代码版本库对应的版本分支里, ...

  7. 如何通过友盟分析发布后App崩溃日志

    http://blog.csdn.net/totogo2010/article/details/39892467 要分析崩溃日志,首先需要保留发布时的编译出来的.xcarchive文件.这个文件包含了 ...

  8. 友盟错误日志分析(转自:COCOACHINA shemy )

      在做的项目中,用到了友盟的组件,在没有禁用错误日志上传之前,收集了一些错误日志. 有一些朋友看到了错误日志,却不知道怎么定位到程序的的代码中,实际上,这一步是非常的简单.友盟没有集成.dSYM文件 ...

  9. 打造高仿QQ的友盟反馈界面(MVP模式)

    什么是MVP呢,简单来说就是将view层和逻辑完全独立出来,让逻辑和显示完全独立.本例中就是采用了这种模式,让activity作为view层,activity中涉及了适配器,所以这里尝试让适配器作为P ...

随机推荐

  1. 游戏搭服自动化脚本shell

    #!/bin/bash #;g=6006c= 178pop_s10 rm-bp1gy2r82o607w4v8.mysql.rds.aliyuncs.com # basedir=/data/jzadmi ...

  2. maven构建SSM框架中pom.xml文件配置

    <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/20 ...

  3. [PAClient Error] Error: E4356 File does not exist armv7

    [PAClient Error] Error: E4356 File does not exist: /Users/tt/PAServer/scratch-dir/Administrator-snIO ...

  4. phpstudy+dvwa配置

    1.Apache端口被系统进程占用(Pid一般为4) -- 修改apache端口,然后通过localhost:端口号访问 2.DVWA System error - config file not f ...

  5. 项目(五)jumpserver企业开源跳板机搭建

    跳板机是什么?跳板机是运维堡垒主机的另个称呼.作为技术或者运维人员应该不会陌生.企业为了服务器的安全,通常所有的ssh连接都是通过跳板机来完成,以便于对ssh连接进行验证和管理. 接下来,我来讲述一下 ...

  6. SpringJDBC中jdbcTemplate 的使用

    一:定义 SpringJDBC是spring官方提供的一个持久层框架,对JDBC进行了封装,提供了一个JDBCTemplated对象简化JDBC的开发.但Spring本身不是一个orm框架,与hibe ...

  7. python虚拟环境的搭建

    使用python虚拟环境作用是项目与项目之间相互隔离,互相不受影响,比如当需要同时部署A.B两个项目时,A项目依赖C库的1.0版本,B项目依赖C库的2.0版本,假如不使用虚拟环境隔离A项目和B项目就很 ...

  8. ppt复制文本框文字到word的方法

    打开ppt按Alt+F11,插入--模块,  选中“工具”--“引用”--MicroSoft Word .. 复制代码: Sub Main() On Error Resume Next Dim tem ...

  9. 给对话框添加动画 Dialog

    先添加一个动画文件(res->anim文件夹中),文件名为a.xml <?xml version="1.0" encoding="utf-8"?&g ...

  10. [leetcode]44. Wildcard Matching万能符匹配

    Given an input string (s) and a pattern (p), implement wildcard pattern matching with support for '? ...