发一个异步图片加载控件。网上也有大把的异步网络加载图片的控件,但是有一个问题,异步加载会造成列表中的图片混乱,因为列表的每一项的View都可能被重用,异步加载的时候多个异步线程引用到了同一个View造成图片加载混乱。该控件解决这个问题:
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.URL;
import java.net.URLConnection;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.drawable.Drawable;
import android.net.Uri;
import android.os.AsyncTask;
import android.util.AttributeSet;
import android.widget.ImageView;
/**
* 异步图片控件
* 使用:new AsyncImageView().asyncLoadBitmapFromUrl("http://xxxx","缓存路径"){
*
* @author gaoomei@gmail.com
* @site http://obatu.sinaapp.com
* @version 1.0
* @2011-12-3
*/
public class AsyncImageView extends ImageView {
/**
* 异步task加载器
*/
private AsyncLoadImage mAsyncLoad;
/**
* 下载回来的图片缓存存活时间,单位:秒(s),默认30分钟
*/
private long mCacheLiveTime = 1800;
public AsyncImageView(Context context) {
super(context);
}
public AsyncImageView(Context context, AttributeSet attrs) {
super(context, attrs);
}
public AsyncImageView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
/**
*
*/
@Override
public void setImageDrawable(Drawable drawable) {
if (mAsyncLoad != null) {
mAsyncLoad.cancel(true);
mAsyncLoad = null;
}
super.setImageDrawable(drawable);
}
/**
* 重写下面几个设置图片资源的方法,目地是取消网络加载
*/
@Override
public void setImageResource(int resId) {
cancelLoad();
super.setImageResource(resId);
}
@Override
public void setImageURI(Uri uri) {
cancelLoad();
super.setImageURI(uri);
}
@Override
public void setImageBitmap(Bitmap bitmap) {
cancelLoad();
super.setImageBitmap(bitmap);
}
/**
* 取消正在进行的异步task
*/
public void cancelLoad() {
if (mAsyncLoad != null) {
mAsyncLoad.cancel(true);
mAsyncLoad = null;
}
}
/**
* 设置图片存活时间
*
* @param second
* 存活时间,单位【秒】,如果等于0或null,则不缓存
*/
public void setCacheLiveTime(long second) {
if (second == 0) {
this.mCacheLiveTime = 0;
} else if (second >= 0) {
this.mCacheLiveTime = second * 1000;
}
}
/**
* 从网络异步加载
*
* @param url
* @param saveFileName
*/
public void asyncLoadBitmapFromUrl(String url, String saveFileName) {
if (mAsyncLoad != null) {
mAsyncLoad.cancel(true);
}
// AsyncTask不可重用,所以每次重新实例
mAsyncLoad = new AsyncLoadImage();
mAsyncLoad.execute(url, saveFileName);
}
/**
* 异步加载器
*/
private class AsyncLoadImage extends AsyncTask<String, Integer, Bitmap> {
/**
* 是否取消
*/
private boolean isCancel = false;
@Override
protected Bitmap doInBackground(String... params) {
if (isCancel) {
return null;
}
String url = params[0];
String fileName = params[1];
try {
return getBitmap(url, fileName);
} catch (IOException e) {
e.printStackTrace();
}
return null;
}
@Override
protected void onCancelled() {
System.out.println("async load imgae cancel");
isCancel = true;
}
@Override
protected void onPostExecute(Bitmap result) {
if (!isCancel && result != null) {
AsyncImageView.this.setImageBitmap(result);
}
}
}
/**
* 下载图片
*
* @param urlString
* url下载地址
* @param fileName
* 缓存文件路径
* @throws IOException
*/
private Bitmap getBitmap(String urlString, String fileName)
throws IOException {
if (fileName == null || fileName.trim().isEmpty()) {
InputStream input = getBitmapInputStreamFromUrl(urlString);
return BitmapFactory.decodeStream(input);
}
File file = new File(fileName);
if (!file.isFile()
|| (mCacheLiveTime > 0 && (System.currentTimeMillis()
- file.lastModified() > mCacheLiveTime))) {
InputStream input = getBitmapInputStreamFromUrl(urlString);
file = saveImage(input, fileName);
// 如果文件结构创建失败,则直接从输入流解码图片
if (file == null || !file.exists() || !file.canWrite()
|| !file.canRead()) {
return BitmapFactory.decodeStream(input);
}
}
return BitmapFactory.decodeFile(file.getAbsolutePath());
}
/**
* 下载图片,输入InputStream
*
* @param urlString
* @return
* @throws IOException
*/
private InputStream getBitmapInputStreamFromUrl(String urlString)
throws IOException {
URL url = new URL(urlString);
URLConnection connection = url.openConnection();
connection.setConnectTimeout(25000);
connection.setReadTimeout(90000);
return connection.getInputStream();
}
/**
* 从输入流保存图片到文件系统
*
* @param fileName
* @param input
* @return
*/
private File saveImage(InputStream input, String fileName) {
if (fileName.trim().isEmpty() || input == null) {
return null;
}
File file = new File(fileName);
OutputStream output = null;
try {
file.getParentFile().mkdirs();
if (file.exists() && file.isFile()) {
file.delete();
}
if (!file.createNewFile()) {
return null;
}
output = new FileOutputStream(file);
byte[] buffer = new byte[4 * 1024];
do {
// 循环读取
int numread = input.read(buffer);
if (numread == -1) {
break;
}
output.write(buffer, 0, numread);
} while (true);
output.flush();
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
output.close();
} catch (IOException e) {
e.printStackTrace();
} catch (Exception e2) {
e2.printStackTrace();
}
}
return file;
}
}
来自:
obatu
分享到:
相关推荐
在`LvAdapter`这个项目中,我们可以期待看到上述概念的实现,这将是一个实用的学习资源,帮助理解如何创建高效、可重用的自定义Adapter。对于需要从Eclipse迁移到Android Studio的开发者,主要工作在于转换项目结构...
在`getView()`方法中,我们可以根据数据项的内容设置视图中的各个组件,例如TextView、ImageView等。同时,这个方法也是添加动画、点击事件等交互功能的地方。 接下来,我们来看一下如何在ListView中使用自定义...
最后,别忘了在自定义Adapter中实现其他必要的方法,如getItemCount()和getItemId(),以便正确地处理数据源和ListView的交互。 通过以上步骤,我们成功地运用了ViewHolder优化了自定义Adapter。这种方法显著提高了...
在getView()方法中,我们首先检查convertView是否为空,如果不为空,就直接复用这个视图,避免每次都创建新的View对象。通过设置View的各个部分(如TextView, ImageView等)来显示对应的数据。这样,当有大量数据时...
自定义Adapter的核心在于`getView()`方法,这里我们将重点讨论如何在这个方法中实现图片上显示数字的功能。通常,`getView()`会接收一个 convertView 参数,这是一个可重用的视图,目的是提高性能。如果 convertView...
convertView是`getView()`方法中的一个参数,它是已存在的视图,可以被复用来显示新的数据,从而避免了频繁创建新视图导致的性能问题。当一个item滚动出屏幕后,它的视图可以被复用为即将进入屏幕的其他item,这样就...
1. 在适配器的getView()方法中,确保每次对convertView进行重用时,都先清空并重新设置所有视图的内容。避免简单地覆盖数据,而应确保每个视图元素都被正确地初始化。 2. 使用“if (convertView == null)”判断,只...
在`getView()`方法中,我们需要检查convertView是否为空,如果不为空,我们可以直接复用,这样可以大大提高效率,减少内存分配和回收的操作。 在"MultiItemListView"这个Demo中,可能会包含以下关键部分: 1. ...
在自定义AdapterView时,你可能需要关注`onScrollStateChanged()`和`onScroll()`方法,它们在滚动状态改变或实际滚动发生时被调用。如果你的自定义AdapterView需要支持滚动,那么理解和实现这两个方法至关重要。 ...
当ListView滚动时,不在屏幕范围内的item视图会被回收,并作为convertView传递给`getView()`方法。开发者需要根据数据和视图的状态来复用或重新配置这个视图。 2. getView()方法的作用: ListView的Adapter中的`...
7. **性能优化**:为了防止内存泄漏和提高性能,记得在Adapter的getView()方法中重用convertView,如果可能的话,还要避免在该方法中执行耗时的操作。 8. **自定义LayoutManager**(适用于RecyclerView):虽然原生...
1. 在自定义Adapter的getView()方法中,首先检查convertView是否为空。如果不为空,那么需要清空该视图,避免旧数据的残留。可以使用`convertView.setTag(null)`和`convertView.removeAllViews()`来清除。 2. 接着,...
在Android开发中,Adapter是一个非常重要的组件,它用于在数据集和UI视图之间建立桥梁,使得数据可以被适配并展示在ListView、GridView、RecyclerView等控件中。本篇文章将详细探讨“最简洁的Adapter”及其在Android...
这个方法会根据传入的position(当前项在数据源中的位置)和convertView(可重用的视图,为提高性能)来创建或复用View。 - 更新:当数据源发生变化时,可以通过调用Adapter的`notifyDataSetChanged()`方法通知...
在这个自定义Adapter中,我们需要重写以下方法: 1. `getCount()`:返回数据源的大小。 2. `getItem(int position)`:根据位置获取数据源中的项。 3. `getItemId(int position)`:返回项的唯一ID,通常用位置作为ID...
- 优化性能:避免在getView()方法中进行耗时操作,如网络请求、图片解码等,这可能导致ListView滚动不流畅。可以使用AsyncTask或其他异步方式处理这些任务。 3. **源码解析** ListView的底层实现涉及到了...
3. **实现数据绑定**:在Adapter的`getView()`方法中,将数据项与视图元素进行绑定。通常,为了性能优化,会复用convertView(已存在的视图)。 4. **数据更新**:当数据源发生变化时,需要调用Adapter的相关方法...
4. 自定义ListView的外观:在自定义适配器的`getView()`方法中,我们可以通过获取传入的convertView并设置其内容来定制每个列表项的外观。如果convertView为null,我们需要创建一个新的视图;否则,我们可以重用它以...
然而,在ListView中嵌套多个EditText时,常常会遇到一个问题,即用户在EditText中输入内容时,ListView滚动时导致显示混乱,原本输入的位置与EditText不匹配,这被称为“EditText焦点错乱”或“ListView复用机制问题...