为什么会乱序
(1)listview中的imageview控件数比一屏显示的控件数量稍微多一点,移除屏幕的imageviw控件会进入到recycleBin(用来缓存用过的view)中,新进入屏幕的元素会从recycleBin中获取imageview控件。
(2) 每当有新元素进入屏幕的时候会调用getview()方法,此方法中会开启异步请求从网络上获取图片,网络操作比较耗时,当我们快速滑动listview,某一个位置正在加载图片,还没有加载完成就被移出了屏幕。根据listview的工作原理,被移出屏幕的控件会被新进入屏幕的元素利用,而刚好前面的图片请求有了响应,就会将刚才位置上的图片显示到现在位置上,虽然他们位置不同,但都是引用的同一个imageview实例,也就出现了乱序的情况。
(3) 新进入屏幕的的元素又会发起新的请求获取图片,并显示在当前位置的imageview上,所以就会出现先显示一张图片,接着再显示另一张图片的情况。
如何解决
1.findViewWithTag()
先调用控件的setTag()方法给控件设置一个标签,然后在调用listview的findViewWithTag()方法使用相同Tag名来找回控件。
由于listView中的ImageView控件都是重用的,被移出屏幕的控件很快就会进入的图片重新利用,那么getView()方法就会被调用,而我们在getView方法中会为imageview控件设置新的Tag,老的Tag就会被覆盖掉,于是这是在调用findViewWithTag()方法并传入老的Tag得到的只会是null,而我们判断只有imageview不等于null的时候才会对它设置图片,这样图片乱序的问题就会解决。
2.使用NetworkImageView
NetworkImageView是Volley当中提供的控件,使用它要在XML布局中。
3.使用弱引用关联
本质是要让ImageView和BitmapWorkerTask之间建立一个双向关联,互相持有对方的引用,再通过适当的逻辑判断来解决图片乱序问题,然后为了防止出现内存泄漏的情况,双向关联要使用弱引用的方式建立。