How to load the Listview “smoothly” in android

前端 未结 5 2017
盖世英雄少女心
盖世英雄少女心 2020-11-30 20:09

I load data from Cursor to listview, but my Listview not really display \"smooth\". The data change when I drag up and down on the scollbar in my ListView. And some items lo

相关标签:
5条回答
  • 2020-11-30 20:11

    I will describe you how to get such issue that you have. Possibly this will help you.

    So, in list adapter you have such code:

    public View getView(int position, View contentView, ViewGroup arg2)
        {
            ViewHolder holder;
    
            if (contentView == null) {
                holder = new ViewHolder();
                contentView = inflater.inflate(R.layout.my_magic_list,null);
                holder.label = (TextView) contentView.findViewById(R.id.label);
                contentView.setTag(holder);
            } else {
                holder = (ViewHolder) contentView.getTag();
            }
    
            holder.label.setText(getLabel());
    
            return contentView;
        }
    

    As you can see, we set list item value only after we have retrieved holder.

    But if you move code into above if statement:

    holder.label.setText(getLabel());
    

    so it will look after like below:

    if (contentView == null) {
       holder = new ViewHolder();
       contentView = inflater.inflate(R.layout.my_magic_list,null);
       holder.label = (TextView) contentView.findViewById(R.id.label);
       holder.label.setText(getLabel());
       contentView.setTag(holder);
    }
    

    you will have your current application behavior with list item duplication.

    Possibly it will help.

    0 讨论(0)
  • 2020-11-30 20:12

    ListView is a tricky beast.

    Your second question first: you're seeing duplicates because ListView re-uses Views via convertView, but you're not making sure to reset all aspects of the converted view. Make sure that the code path for convertView!=null properly sets all of the data for the view, and everything should work properly.

    You'll want your getView() method to look roughly like the following if you're using custom views:

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        final MyCustomView v = convertView!=null ? (MyCustomView)convertView : new MyCustomView();
        v.setMyData( listAdapter.get(position) );
        return v;
    }
    

    If you're not using your own custom view, just replace the call to new MyCustomView() with a call to inflater.inflate(R.layout.my_layout,null)

    As to your first question, you'll want to watch Romain's techtalk on ListView performance here: http://code.google.com/events/io/sessions/TurboChargeUiAndroidFast.html

    From his talk and in order of importance from my own experience,

    • Use convertView
    • If you have images, don't scale your images on the fly. Use Bitmap.createScaledBitmap to create a scaled bitmap and put that into your views
    • Use a ViewHolder so you don't have to call a bunch of findViewByIds() every time
    • Decrease the complexity of the views in your listview. The fewer subviews, the better. RelativeLayout is much better at this than, say, LinearLayout. And make sure to use if you're implementing custom views.
    0 讨论(0)
  • 2020-11-30 20:30

    You can see the reocurring text in multiple rows if you handle it in the wrong way. I've blogged a bit about it recently - see here. Other than that you might want to take a look at ListView performance optimization. Generally it's because of the view reuse and I've seen it few times already.

    0 讨论(0)
  • 2020-11-30 20:32

    I'm facing this problem as well, but in my case I used threads to fetch the external images. It is important that the current executing thread do not change the imageView if it is reused!

    public View getView(int position, View vi, ViewGroup parent) {
    ViewHolder holder;
    String imageUrl = ...;
    
    if (vi == null) {
        vi = inflater.inflate(R.layout.tweet, null);
        holder = new ViewHolder();
        holder.image = (ImageView) vi.findViewById(R.id.row_img);
        ...
        vi.setTag(holder);
    } else {
        holder = (ViewHolder) vi.getTag();      
    }
    holder.image.setTag(imageUrl);
    ...
    DRAW_MANAGER.fetchDrawableOnThread(imageUrl, holder.image);
    }
    

    And then on the fetching thread I'm doing the important check:

    final Handler handler = new Handler() {
    @Override
    public void handleMessage(Message message) {
         // VERY IMPORTANT CHECK
        if (urlString.equals(url))
            imageView.setImageDrawable((Drawable) message.obj);
    };
    
    Thread thread = new Thread() {  
    
    @Override
    public void run() {
        Drawable drawable = fetchDrawable(urlString);
        if (drawable != null) {
            Message message = handler.obtainMessage(1, drawable);
            handler.sendMessage(message);
        }
    }};
    thread.start();
    

    One could also cancel the current thread if their view is reused (like it is described here), but I decided against this because I want to fill my cache for later reuse.

    0 讨论(0)
  • 2020-11-30 20:34

    Just one tip: NEVER use transparent background of item layout - it slows performance greatly

    0 讨论(0)
提交回复
热议问题