0

我很难实现延迟加载列表。我知道,关于类似问题的帖子数不胜数,但请在判断之前先看看我的问题。

目标:加载 SD 卡上存在的位图缩略图,但不阻塞主 (UI) 线程,因此列表将平滑滚动。

资料来源:我尝试的基础是这两个帖子:

到目前为止我的尝试

我试图实现一个类来加载和缓存(我不知道,如果它是适当的表达式)缩略图。它看起来像这样(我尽量不贴一堵代码墙,所以我删掉了不重要的部分):

public class ThumbnailContainer
{
    //this will store the thumbnails 
    private final HashMap<File, SoftReference<Bitmap>> data;

    //this Handler class will update the ui, when we got the thumb from a thread
    private static final class BitmapHandler extends Handler
    {
        private final WeakReference<ImageView> image;
        public BitmapHandler(ImageView image) 
            {this.image = new WeakReference<ImageView>(image);}

        @Override
        public void handleMessage(Message msg)
        {
            if(image.get()!=null)
                image.get().setImageBitmap((Bitmap) msg.obj);
        }
    }

    public ThumbnailContainer(Context context)
    {
        data = new HashMap<File, SoftReference<Bitmap>>();
    }

    //this will set the Bitmap to the ImageView (load on a thread if required)
    public void setBitmapOnThread(final File file, final ImageView view)
    {
        //contains will return true, if the data map contains the file key
        //and the SoftReference is still vaild.
        if (contains(file))
        {
            view.setImageBitmap(data.get(file).get());
            return;
        } 
        else
        {
            final Handler handler = new BitmapHandler(view);

            final Thread thread = new Thread()
            {
                @Override
                public void run()
                {
                    Bitmap bitmap = getMeasuredBitmap(file);
                    Message msg = handler.obtainMessage(0, bitmap);
                    handler.sendMessage(msg);
                }
            };

            thread.start();
        }
    }

    // load the Bitmap if it isn't already, scales it down, and recycles the original
    private Bitmap getMeasuredBitmap(File file)
    {
        if (contains(file))
            return data.get(file).get();
        else
        {
            Bitmap bitmap = BitmapFactory.decodeFile(file.getAbsolutePath());

            /*counting the scale of the new Bitmap, i cut the detail*/ 

            Bitmap measured = Bitmap.createScaledBitmap(bitmap, w, h, false);
            bitmap.recycle();

            data.put(file, new SoftReference<Bitmap>(measured));

            return measured;
        }
    }

    //returns true, if the map contains this file, and the reference is still valid
    private boolean contains(File file)
    {
        if (data.containsKey(file))
        {
            if (data.get(file).get() != null) return true;
            else return false;
        }
        return false;
    }
}

结果:我仍然得到一个非常滞后的列表滚动。就像我什至没有添加线程解决方案一样,只是在 listadaptersgetView()方法中加载了缩略图。我尝试将Threads优先级(参见setBitmapOnThread())设置为LOW,当我这样做时,滚动通常很流畅,我可以看到缩略图正在加载,但是当我滚动得非常快时,我的内存就会用完。我认为这是因为启动了太多线程,而它们无法完成。

我的问题:你们在这里看到明显的错误吗?

如果不是,那么坚持使用低优先级线程解决方案是否明智?如果是这样,那么有没有办法将线程数限制为固定数量(如 5-6),如果达到最大线程数,在开始新的线程之前停止并加入未完成的?我读过关于 ThreadPools 的文章,但我从未使用过。

我真的很感激任何帮助!

4

1 回答 1

1

我会停止像你一样使用线程并使用 AsyncTask。使用此任务从磁盘获取文件并将其放置在缓存中(如果不存在)。将缓存的值返回到适配器中的 ImageView。

因此,在您的适配器中将 ImageView 传递给 AsyncTask。让任务完成工作。

于 2012-09-10T19:52:13.943 回答