上周完成的工作主要是修改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