fairly new Android developer here.
I've come across a strange problem that I'm not sure how to work around. I've read a lot of problems around here that sound like they are the same problem I am having, but the solutions to their problems never seemed applicable to what is going on here.
I have a custom listView set up using a separate class that extends BaseAdapter and uses a view holder (I set this up looking at various different examples of custom listviews). It's got a text view, a button, a progress bar, and an image button, and these items are hidden/changed based on an AsyncTask used to download a file.
Everything works fine when I scroll through the list slowly. When I scroll the list rapidly up/down, it is as if the views of some cells in the list get re-assigned to other cells.
I put this line into the top of my "getView" method in my adapter class:
@Override
public View getView(final int pos, View convertView, ViewGroup parent)
{
Log.i("ks3","getView called, position is " + pos);
I have 7 items in my list, and 6 of them fit on screen. When it first gets displayed, I see this in my log:
getView called, position is 0
getView called, position is 1
getView called, position is 2
getView called, position is 3
getView called, position is 4
getView called, position is 5
When I scroll down slowly to the next item, this is printed out next:
getView called, position is 6
When I scroll back up, this:
getView called, position is 0
Going down and up slowly produce these results flawlessly.
When I start scrolling back and forth quickly, it prints out this message a bunch of times, mostly showing 6 and 0 as the position, but occasionally I will see these as well:
getView called, position is 1
getView called, position is 5
Positions 1 and 5 should not be getting called, as they are always on-screen. It is as if getView got all jumbled up. At the same time, as I stated before, my list will look strange. Image buttons will be moved up or down a cell where they should not be.
If I go back to scrolling smoothly, I see only 0's and 6's for position again.
I honestly am not sure how to work around this. I thought maybe I could limit how fast you are able to scroll through the list, but haven't been able to find anything that works.
Thanks!
Edit: I wanted to update a couple things regarding this question. The first being that, from a comment here and from the Google video on listView, it has come to my attention that getView can be called for other things then what I imagined it was for, such as measurements, and so I should not be alarmed by what I originally thought was part of my problem (that being that I thought getView was being called with the wrong positions).
Secondly, I saw multiple times that it is a very bad idea to "cache views inside your adapter." I am not exactly clear on what this means, but I'm pretty sure it is one of the things I am doing wrong (I save an instance of a button, progressBar, imageButton...).
That, along with the fact that I update the views outside of getView and I don't use notifyDataSetChanged(); those together are probably causing some wonky stuff to happen.
You are correct that ListView
is reusing views in different places on the screen. It's an optimization to keep memory use reasonable and speedy by not allocating new views all the time.
Chances are that you're using LiewView
incorrectly. Watch this talk on how to properly use ListView
to get the whole story, but here's the highlights:
getView()
method or you will get strange behavior.getView(int, View, ViewGroup)
provides a view instance, populate its fields instead of inflating totally new views. Assuming you've correctly implemented getItemType()
, you'll always get the right View
type to repopulate. getView()
method as fast as you possibly can, and only do heavy lifting on other threads.getView()
doesn't necessarily mean the data will be displayed. The framework uses these for measurement purposes. Since the work could be thrown away, this is another reason to make sure that getView()
is as fast as you can make it.notifyDataSetChanged()
. Don't fiddle with the views directly, they'll be populated on the next UI loop when it gets redrawn.Having just spent a few days reworking a ListView
that was implemented naively, I feel your pain. The results have been worth it, though!
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With