我有一个listadapter,我有很多listitems,每个项目都包含textview和imageview。我想从后端下载图像(以Base64编码),所以Picasso和via URL 下载解决方案对我不利。 在我下载Base64解码并创建位图之后,我只是坚持如何在列表适配器中启动后台线程,其中视图始终在回收(当用户滚动时)。
我开始创建一个带有Hanlder的线程,其中处理程序将位图插入到imageview中,但它没有工作(在内存中)。之后我发了一个AsyncTask,但在这种情况下我有回收问题,namley:当用户向下滚动时,我在底部视图中看到顶部图像:(
使用改装下载
你能帮帮我吗?我的AsyncTaks代码: 这里的问题是,我更新了imageview,但也许用户已经滚动了!
public DownloadImageAsyncTask(ImageView iw, Display display, String imageID, ImagesCache imagesCache) { this.iw = iw; this.imageCache = imagesCache; this.imageID = imageID; this.display = display; }
@Override
protected Void doInBackground(Void... params) {
String[] imageArray = new String[1];
imageArray[0] = imageID;
ImageResponse imageResponse = new IdeaBackend().getImageByID(imageArray);
bitmap = UserExperienceHelper.decodeBase64AndScaleDownImage(imageResponse.getResponse().get(0).getImageBase64Code(), display);
imageCache.put(imageID, bitmap);
return null;
}
@Override
protected void onPostExecute(Void aVoid) {
super.onPostExecute(aVoid);
this.iw.setImageBitmap(bitmap);
}
答案 0 :(得分:0)
您需要回收位图或视图。可以轻松处理列表视图的增量加载。这是一篇可能对你有帮助的帖子!
答案 1 :(得分:0)
在适配器中:
View listItemLayout = (View) inflater.inflate(...);
ImageView imageView = (ImageView) listItemLayout.findViewById(R.id.imageView);
MyImageLoader.loadImage(activity, imageView, id);
这是公共课,将为您下载和缓存图像。
public class MyImageLoader {
private HashMap < String, Bitmap > cache = new HashMap < String, Bitmap > ();
public static loadImage(final Activity activity, final ImageView imageView, final String id) {
new Thread(new Runnable() {
@Override
public void run() {
if (cache.containsKey(id) {
//Bitmap already exists.
} else {
//This bitmap has not been downloaded so you need to download the bitmap.
//Once bitmap is downloaded, add it to the HashMap.
}
activity.runOnUiThread(new Runnable() {
public void run() {
imageView.setImageBitmap(cache.get(id));
}
});
}
}).start();
}
}
现在,无论何时imageView.setImageBitmap()
,图像都会自动更新。您无需单独刷新视图。
对上一代码的补充:
public class MyImageLoader {
private static HashMap <String, Bitmap> cache = new HashMap <String, Bitmap>();
private static HashMap<String, DownloadListener> listeners = new HashMap()<>;
private static ArrayList<String> currentlyDownloading = new ArrayList()<>;
public static loadImage(final Activity activity, final ImageView imageView, final String id) {
DownloadListener downloadListener = new DownloadListener(){
@Override
public void onDownloadComplete(){
activity.runOnUiThread(new Runnable() {
public void run() {
imageView.setImageBitmap(cache.get(id));
}
});
}
@Override
public void onDownloadFailed(){
// Do something when download fails.
}
};
if(!listeners.containsKey(id)){
listener.put(id, new ArrayList<DownloadListener>());
}
listener.get(id).add(downloadListener);
if(cache.containsKey(id)){
for(DownloadListener listener : listeners.get(id)){
listener.onDownloadComplete();
}
} else if(!currentlyDownloading.contains(id)) {
new Thread(new Runnable() {
@Override
public void run() {
currentlyDownloading.add(id);
//Start Downloading
//Download finished
currentlyDownloading.remove(id);
//If download successful
cache.put(id, bitmap);
for(DownloadListener listener : listeners.get(id)){
listener.onDownloadComplete();
}
/*
If download failed
for(DownloadListener listener : listeners.get(id)){
listener.onDownloadFailed();
}
*/
currentlyDownloading.remove(id);
}
}).start();
}
}
public interface DownloadListener {
public void onDownloadComplete();
public void onDownloadFailed();
}
}
答案 2 :(得分:0)
首先,我要感谢您对解决方案的意见和想法。 在我问你的时候,我在过去的两天里正在研究这个问题,所以实际上我提出了我的最终解决方案,当我在这里发布时已经部分准备好了:
public class IdeaAdapter extends ArrayAdapter<Idea> {
private final Context context;
private final Display display;
private List<Idea> dataList;
private Set<String> imagesInProgress;
public IdeaAdapter(Display display, Activity c, List<Idea> dataList) {
super(c, android.R.layout.simple_list_item_1, dataList);
this.dataList = dataList;
this.context = c;
this.display = display;
imagesInProgress = new HashSet<>();
}
@Override
public int getCount() {
return dataList.size();
}
@Override
public Idea getItem(int position) {
return dataList.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(final int position, View convertView, ViewGroup parent) {
final ViewHolder vh;
if (convertView == null) {
vh = new ViewHolder();
convertView = LayoutInflater.from(context).inflate(R.layout.idea_list_item, null);
vh.ideaImage = (ImageView) convertView.findViewById(R.id.idea_image);
vh.ideaName = (TextView) convertView.findViewById(R.id.idea_title);
vh.progressWheel = (ProgressWheel) convertView.findViewById(R.id.progress_wheel);
convertView.setTag(vh);
} else {
vh = (ViewHolder) convertView.getTag();
}
final Idea ideaItem = dataList.get(position);
vh.ideaName.setText(ideaItem.getTitle());
if (ideaItem.getImages().size() > 0) {
final String imageId = ideaItem.getImages().get(0);
Log.e(" isContains", String.valueOf(ImagesCache.getInstance().containsKey(String.valueOf(imageId))));
if (!ImagesCache.getInstance().containsKey(String.valueOf(imageId)) && !imagesInProgress.contains(imageId)) {
Log.e(" downloading", String.valueOf(imageId));
vh.ideaImage.setImageBitmap(BitmapFactory.decodeResource(context.getResources(), R.drawable.ic_launcher));
imagesInProgress.add(imageId);
new DownloadImageAsyncTask(vh.ideaImage, display, imageId).execute();
} else {
Log.e("CACHE", String.valueOf(imageId));
vh.ideaImage.setImageBitmap(ImagesCache.getInstance().getBitmap(String.valueOf(imageId)));
}
}
return convertView;
}
private static class ViewHolder {
private ImageView ideaImage;
private TextView ideaName;
private ProgressWheel progressWheel;
}
}
图像缓存:
public class ImagesCache {
private static ImagesCache INSTANCE;
private static Context context;
private static DiskLruCache mDiskCache;
private static Bitmap.CompressFormat mCompressFormat = Bitmap.CompressFormat.JPEG;
private static int mCompressQuality = 70;
private static final int APP_VERSION = 1;
private static final int VALUE_COUNT = 1;
private static final String TAG = "DiskLruImageCache";
private static final String uniqueName = "IdeasCache";
private static final int diskCacheSize = 20*1024*1024; // in bytes
/**
* Make sure to give an application context to me.
* I like to keep it as a static reference
*
* @param c
*/
public static void init(Context c) {
context = c;
}
public static ImagesCache getInstance() {
if (INSTANCE == null) {
INSTANCE = new ImagesCache();
}
return INSTANCE;
}
private ImagesCache() {
try {
final File diskCacheDir = getDiskCacheDir(context, uniqueName);
mDiskCache = DiskLruCache.open(diskCacheDir, APP_VERSION, VALUE_COUNT, diskCacheSize);
/*mCompressFormat = compressFormat;
mCompressQuality = quality;*/
} catch (IOException e) {
e.printStackTrace();
}
}
private static boolean writeBitmapToFile(Bitmap bitmap, DiskLruCache.Editor editor)
throws IOException {
OutputStream out = null;
try {
out = new BufferedOutputStream(editor.newOutputStream(0), Utils.IO_BUFFER_SIZE);
return bitmap.compress(mCompressFormat, mCompressQuality, out);
} finally {
if (out != null) {
out.close();
}
}
}
private File getDiskCacheDir(Context context, String uniqueName) {
// Check if media is mounted or storage is built-in, if so, try and use external cache dir
// otherwise use internal cache dir
final String cachePath =
Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState()) ||
!Utils.isExternalStorageRemovable() ?
Utils.getExternalCacheDir(context).getPath() :
context.getCacheDir().getPath();
return new File(cachePath + File.separator + uniqueName);
}
public static void put(String key, Bitmap data) {
DiskLruCache.Editor editor = null;
try {
editor = mDiskCache.edit(key);
if (editor == null) {
return;
}
if (writeBitmapToFile(data, editor)) {
mDiskCache.flush();
editor.commit();
if (BuildConfig.DEBUG) {
Log.d("cache_test_DISK_", "image put on disk cache " + key);
}
} else {
editor.abort();
if (BuildConfig.DEBUG) {
Log.d("cache_test_DISK_", "ERROR on: image put on disk cache " + key);
}
}
} catch (IOException e) {
if (BuildConfig.DEBUG) {
Log.d("cache_test_DISK_", "ERROR on: image put on disk cache " + key);
}
try {
if (editor != null) {
editor.abort();
}
} catch (IOException ignored) {
}
}
}
public static Bitmap getBitmap(String key) {
Bitmap bitmap = null;
DiskLruCache.Snapshot snapshot = null;
try {
snapshot = mDiskCache.get(key);
if (snapshot == null) {
return null;
}
final InputStream in = snapshot.getInputStream(0);
if (in != null) {
final BufferedInputStream buffIn =
new BufferedInputStream(in, Utils.IO_BUFFER_SIZE);
bitmap = BitmapFactory.decodeStream(buffIn);
}
} catch (IOException e) {
e.printStackTrace();
} finally {
if (snapshot != null) {
snapshot.close();
}
}
if (BuildConfig.DEBUG) {
Log.d("cache_test_DISK_", bitmap == null ? "" : "image read from disk " + key);
}
return bitmap;
}
public static boolean containsKey(String key) {
boolean contained = false;
DiskLruCache.Snapshot snapshot = null;
try {
snapshot = mDiskCache.get(key);
contained = snapshot != null;
} catch (IOException e) {
e.printStackTrace();
} finally {
if (snapshot != null) {
snapshot.close();
}
}
return contained;
}
public static void clearCache() {
if (BuildConfig.DEBUG) {
Log.d("cache_test_DISK_", "disk cache CLEARED");
}
try {
mDiskCache.delete();
} catch (IOException e) {
e.printStackTrace();
}
}
public static File getCacheFolder() {
return mDiskCache.getDirectory();
}
}
所有我需要做的,管理线程(阻塞队列)以不为每个getView调用触发!