我很难实现延迟加载列表。我知道,关于类似问题的帖子数不胜数,但请在判断之前先看看我的问题。
目标:加载 SD 卡上存在的位图缩略图,但不阻塞主 (UI) 线程,因此列表将平滑滚动。
资料来源:我尝试的基础是这两个帖子:
- ListView 非常滞后 - Android (ViewHolder 想法)
- ListView 中图像的延迟加载 (线程思想)
到目前为止我的尝试:
我试图实现一个类来加载和缓存(我不知道,如果它是适当的表达式)缩略图。它看起来像这样(我尽量不贴一堵代码墙,所以我删掉了不重要的部分):
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 的文章,但我从未使用过。
我真的很感激任何帮助!