Android - Issue with lazy loading images into a ListView

后端 未结 4 1989
星月不相逢
星月不相逢 2020-12-29 15:18

This is a very common scenario: displaying images in a ListView which have to be downloaded from the internet.

Right now I have a custom subclass of ArrayAdapter whi

相关标签:
4条回答
  • 2020-12-29 15:24

    I had the same issue.

    After almost 2 days of heavy debugging/optimizing and trying to figure out, why my getView() is called for all views over and over again when using setImageBitmap() in a Row, I came up with a dirty solution:

    1) Extend a custom ImageView which you use for all the Images in your List

    2) in this ImageView overwrite the method

    @Override
    public void requestLayout()
    { 
      return; 
    }
    

    3) Dirty, but for me it works

    4) Profit ;)

    0 讨论(0)
  • 2020-12-29 15:32

    In the linked solution, fetchDrawableOnThread() should only be called if the view does not already have the correct drawable.

    A view does not have a drawable if getDrawable() returns null.

    If you are reusing slots, you views you need to go further and manage the state. If your views have a member variable storing the URL, and a boolean to say whether it is loaded, it'd be easy to know whether to call fetchDrawableOnThread() or not, for example.

    I'd speculate that the drawable's toString() detailed the path from which the image was loaded. (If it doesn't, you could subclass the drawable returned to make it so). In this case, you could avoid the boolean outlined above and just do a comparison to determine if its the right drawable or whether to fetch a replacement.

    Additionally, your getView() on a visible row should ensure that those that no longer visible get unloaded, to prevent memory exhaustion. A finesse would be to move the no longer visible images to soft references (so they are unloaded when memory is needed) as another poster on the original thread noted.

    0 讨论(0)
  • 2020-12-29 15:48

    I used the code in following link : another stackoverflow question

    i made small changes in order to solve recycling view problem.i set the url of image to Tag of imageview in the adapter. Following code contains my solution that solves recycling problem:

    public void fetchDrawableOnThread(final String urlString, final ImageView imageView,Drawable drw) {
    
        imageView.setImageDrawable(drw);//drw is default image
        if (drawableMap.containsKey(urlString)) {
            if(imageView.getTag().toString().equals(urlString))
            {
                imageView.setImageBitmap(drawableMap.get(urlString));
                imageView.invalidate();
                return;
            }
    
        }
    
        final Handler handler = new Handler() {
            @Override
            public void handleMessage(Message message) {
                BitmapWrapper wrapper = (BitmapWrapper)message.obj;
                if(wrapper.imageurl.equals(imageView.getTag().toString()))
                {
                    imageView.setImageBitmap((Bitmap)wrapper.bitmap);
                    imageView.invalidate();
                }
    
            }
        };
    
        Thread thread = new Thread() {
            @Override
            public void run() {
                //TODO : set imageView to a "pending" image
    
                Bitmap drawable = fetchDrawable(urlString);
                BitmapWrapper wrapper = new BitmapWrapper();
                wrapper.bitmap = drawable;
                wrapper.imageurl = urlString;
                Message message = handler.obtainMessage(1, wrapper);
                handler.sendMessage(message);
            }
        };
        thread.start();
    }
    
    
        public class BitmapWrapper
    {
        public Bitmap bitmap;
        public String imageurl;
    }
    
    0 讨论(0)
  • 2020-12-29 15:48

    I have a ThumbnailAdapter that wraps up this whole pattern that may help.

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