使用了过滤条件的 ArrayAdapter,向源数据添加或删除数据后刷新时,界面不会同步更新

上周完成的工作主要是修改BUG,当点击清空历史记录按钮时,listview清除全部记录。

问题描述:在ActivityA中的edittext中输入搜索内容,点击搜索按钮后,跳转到B界面,在A的listview中显示符合条件的记录信息。当再次点击搜索框是,又回到A,光标此时在搜索框中,显示键盘,提示输入;同时A的listview中显示刚刚的搜索记录。依次操作,A的listview中item显示所有搜素的记录当光标在edittext中时,点击清空历史记录按钮,listview本应该重新刷新,显示为空的,但是却没有,仍然原先的记录还存在,当点击item时程序出错异常退出。

分析:断点调试,跟踪程序,logcat中打印错误为nullPointException。是因为当点击清空历史记录按钮时,清除数据源,然后adapter.notifyDataSetChanged()。但是界面却没有刷新,但是数据源已经为空了。所以点击记录时会报空指针异常。可是程序中已经调用adapter.notifyDataSetChanged(),问什么界面不刷新呢?

解决办法:

1.  强制listview刷新(失败)

UpdateAllView()、 listView.invalidateViews()、等强制刷新的方法均失效。

2.      强制Activity 刷新(失败)

invaludate()、postInvalidate()。

Invalidate()方法和postInvalidate()都可以在主线程中调用而刷新视图。当Invalidate()被调用的时候,View的OnDraw()就会被调用,Invalidate()必须是在UI线程中被调用,如果在新线程中更新视图的就调用postInvalidate()。

简言之,如果确定是在main thread中调用调用话, 使用 invaludate()
否则要调用 postInvalidate()。调用此种方法后界面可以刷新了,但是用户体验很不好,可以看到界面闪一下,所以放弃。

3.      新建一个空的Adapter(成功)

既然adapter中的数据清空不掉,那么就让listview重新绑定一个空的adapter这样页面就可以刷新清空了。缺点:指标不治本。

4.      解析Adapter源码,分析刷新机制(成功)

mOriginalValues用于过滤数据时保存过滤前的数据,将过滤后的数据存入mObjects。

API17,针对使用了过滤条件的 ArrayAdapter,向源数据添加或删除数据后刷新时,界面不会同步更新。见 https://code.google.com/p/android/issues/detail?id=69179

 原因这是由于删除数据时仅修改了 ArrayAdapter.mOriginalValues 值, 而界面数据来源于ArrayAdapter.mObjects 集合。

 解决方法这里首先清空mOriginalValues 集合,其次应用过滤条件,使得 mObjects 依照 mOriginalValues 更新,保证界面数据被正确刷新。

修改后代码:

history.clear();

adapter.clear();

adapter.getFilter().filter(editText.getText());

这两个是有关这个问题的解释:

https://code.google.com/p/android/issues/detail?id=69179

http://www.cnblogs.com/angeldevil/archive/2011/11/20/2255972.html

这两个链接是TextWatcher的用法的,个人认为较好的。

http://blog.csdn.net/i_lovefish/article/details/7998368

http://www.aichengxu.com/view/40127

 


 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值