Android项目实战-ListView异步图片加载及压缩缓存
这回先讲一下ListView异步加载图片的问题,相关的文章很多,不过在这里我将加载、压缩下载、存储到SD卡等功能全部放上来,方便大家使用或者研究。
1. ListView中的SimpleAdapter中的操作
大家在使用ListView显示复杂的页面时,我们都全重写一个SimpleAdapter,重写其中的getView方法来显示具体内容。相关的内容我们下篇文章再说,这里主要看下图片显示的代码。
//图片内容显示 if(!topic.getImage_value().equals("")){ vh.tl_ll_content_image.setVisibility(View.VISIBLE); //加上标识,要不图片加载会变乱 vh.tl_img_content_image.setTag(topic.getTid()); vh.tl_img_content_image.setImageResource(R.drawable.default_image_load); //新开异步加载图片 LoadingTopicImageAsyncTaskloadingTopicImageAsyncTask = new LoadingTopicImageAsyncTask(vh.tl_img_content_image, topic,ctxContext,true,String.valueOf(topic.getTid())); loadingTopicImageAsyncTask.execute(); }
·vh. tl_ll_content_image是我们在ListView中要显示相关图片的控件。Topic是我们通过getView取到的当前行的数据对象。Topic.getImage_value()中则是要显示的图片url地址,比如:http://www.huoban168.com/huoban/upload/topic/2012/02/weibo_1329196372_1112447.png。
·为图片控件加上setTag主要是防止图片混乱。
·LoadingTopicImageAsyncTask是我们新开一个异步,下载或者加载相关的图片的类。基本上主要的操作都在此类中进行。
2. LoadingTopicImageAsyncTask类中的操作
packagehb.hbwb.asynctask; importhb.hbwb.model.beans.Topic; importhb.hbwb.tools.PicTool; importhb.hbwb.var.PublicVariable; importandroid.app.Activity; importandroid.app.Dialog; importandroid.graphics.Bitmap; importandroid.os.AsyncTask; importandroid.view.View; importandroid.view.View.OnClickListener; importandroid.widget.ImageView; /** * @name 异步加载图片 * @author zhang.yue * @create_date 2011-12-15 * @last_edit_author * @last_edit_date * @remark * @edit_remark */ public classLoadingTopicImageAsyncTask extends AsyncTask<String, Integer,String> { private ImageView imageView = null; private Topic topic = null; private Bitmap bt = null; private Activity activity; private boolean isClick = true; private String tag = ""; /** * 重写构造 * * @paramheaderView * 图像控件 * @paramtopic * @paramactivity * @paramisClick * 是否可以点击放大 */ public LoadingTopicImageAsyncTask(ImageViewimageView, Topic topic, Activity activity, booleanisClick, String tag) { super(); this.imageView = imageView; this.topic = topic; this.activity = activity; this.isClick = isClick; this.tag = tag; } @Override protected String doInBackground(String...params) { // 添加如果是本地缓存的从本地读取... bt =PublicVariable.allTopicImage.get(String.valueOf(topic.getTid())); if (bt == null) { // 图像获取并保存 if (topic.getImage_value().equals("")){ bt = null; } else { bt =PicTool.ReturnBitMap(topic.getImage_value()); } } if (bt!=null) { if(bt.getWidth()>PublicVariable.TOPIC_IMAGE_SHOW-40 ||bt.getHeight()>PublicVariable.TOPIC_IMAGE_SHOW_HEIGHT-40) { bt =PicTool.ChangeSizeBitMap(bt, PublicVariable.TOPIC_IMAGE_SHOW-40); } } return null; } @Override protected void onPostExecute(String result) { super.onPostExecute(result); if (imageView != null &&topic != null && !tag.equals("")) { if(imageView.getTag().toString().equals(tag)) { if (bt != null) { if(isClick) { //点击大图效果 imageView.setOnClickListener(newOnClickListener() { @Override publicvoid onClick(View v) { ShowImageClickListener(bt,activity); } }); } imageView.setImageBitmap(bt); PublicVariable.allTopicImage.put( String.valueOf(topic.getTid()),bt); // 保存到数组中 } } } } /** * 图片点击功能事件处理 * * @parambt * @paramactivity */ public static voidShowImageClickListener(Bitmap bt, Activity activity) { Bitmap maxBt =PicTool.ChangeSizeBitMap(bt, PublicVariable.TOPIC_IMAGE_SHOW); ImageView showImageView = newImageView(activity); showImageView.setImageBitmap(maxBt); //两个嵌套的TabHost中,必须使用activity.getParent() //在首页中使用并无报错,全局使用 Dialog d = newDialog(activity.getParent(), hb.hbwb.R.style.no_back_title_dialog); d.setContentView(showImageView); d.setCanceledOnTouchOutside(true); d.show(); } }
·PublicVariable.allTopicImage是我们写的一个HashMap,存储Bitmap的一个集合,如果在程序的运行过程中多次显示同一个图片即可直接从这里面取出Bitmap并显示,可以节省网上加载或者sd卡读取的步骤,键值是topic的tid,一个微博仅显示一张图片。
·如果上述的Bitmap集合中不存在图片的话,调用PicTool类中的ReturnBitMap返回SD卡上缓存的图片。
·如果图片超过我们规定的一个大小,那么调用PicTool类中的ChangeSizeBitMap方法。
·onPostExecute方法是线程执行完成后,我们将Bitmap对象给予Image图片控件并显示出来。在这里我们还添加了点击事件。
3. ReturnBitMap解析
/** * 返回图片文件用 * * @paramurl * 图片url地址 *@return */ public static Bitmap ReturnBitMap(String url) { String imgName =StringTool.GetImageNameForUrl(url); if (imgName.equals("")) { return null; } else { FileTool ft = newFileTool(); String path =FileFinals.SDCARDROOT + File.separator + FileFinals.SDCARDIMAGEPATH; // 图片文件存在并且不是刷新的时候 if (ft.IsFileExist(imgName +FileFinals.IMAGE_SYSTEM_EXT, FileFinals.SDCARDIMAGEPATH)) { Bitmap bm =ReturnLocalBitMap(path + File.separator + imgName); if(bm==null){ returnReturnWebBitMap(url, imgName, path); }else{ returnbm; } } else { // 先删除原有的 returnReturnWebBitMap(url, imgName, path); } } } /** * 本地获取图片信息 * * @parampath * 图片路径 *@return bitmap对象 */ public static Bitmap ReturnLocalBitMap(Stringpath) { Bitmap bitmap =BitmapFactory.decodeFile(path + ".image"); return bitmap; } /** * 抓取远程图片 * * @paramurl * 图片地址 * @paramimgName * 图片名 * @parampath * 地址 *@return */ public static Bitmap ReturnWebBitMap(Stringurl, String imgName, String path) { Bitmap bitmap = null; try { // 图片大小判断操作 InputStream size_is =HTMLTool.GetHttpConnection(url).getInputStream(); BitmapFactory.Options op =new BitmapFactory.Options(); op.inJustDecodeBounds =true; @SuppressWarnings("unused") Bitmap size_bitmap =BitmapFactory.decodeStream(size_is, null, op); op.inJustDecodeBounds =true; // 仅获取宽高信息,不加载整个图片 boolean isop = false; int size =PublicVariable.TOPIC_IMAGE_SHOW; // 设定获取的大小不能超过宽高 int bili = 0; // 压缩比例 BitmapFactory.Options op_new= new BitmapFactory.Options(); if(op.outWidth>op.outHeight){ if (op.outWidth> size) { bili =op.outWidth / size; isop =true; } }else{ if (op.outHeight> size) { bili =op.outHeight / size; isop =true; } } // 根据比例判断 if (bili != 0) { if (bili < 2) { bili = 2; }else{ bili = bili*2-2; } if(bili%2!=0){ bili =bili+1; } op_new.inSampleSize= bili; } else { isop = false; } System.out.println(bili); size_bitmap = null; size_is.close(); // 如果进行过压缩,加载处理后的图片,如果没有直接加载 InputStream is =HTMLTool.GetHttpConnection(url).getInputStream(); if (isop) { op_new.inPreferredConfig= Bitmap.Config.ARGB_4444; op_new.inPurgeable= true; op_new.inInputShareable= true; bitmap =BitmapFactory.decodeStream(is, null, op_new); } else { bitmap =BitmapFactory.decodeStream(is); } // 将图片写入到内存卡中,做为缓存,将来直接本地读取 WriteBitmapToSdCard(FileFinals.SDCARDIMAGEPATH,path + File.separator + imgName, bitmap); is.close(); } catch (Exception e) { return null; } return bitmap; }
·StringTool.GetImageNameForUrl是截取文件名,比如:http://www.huoban168.com/huoban/upload/topic/2012/02/weibo_1329196372_1112447.png,返回的是weibo_1329196372_1112447.png这一部分的内容。
·接下来查看当前SD卡的缓存文件路径下有没有这张图,如果有的话,直接调用ReturnLocalBitMap加载图片,如果没有的话,调用ReturnWebBitMap方法去调用网络图片并下载下来。
·后缀名都加了一个".image",目的是防止Android的图片浏览器去加载这些图片。
·ReturnWebBitMap中,先通过HTMLConnection获取图片的InputStream二进制流,然后通过 op.inJustDecodeBounds = true; // 这个方法,获取一个只有高度等属性的Bitmap对象,接着根据图片原始大小获取需要大小的压缩比例op_new.inSampleSize = bili;。
·最后取到压缩过的Bitmap对象写入到内存卡或者直接显示均可。至此,图片的下载和保存已完成。
4. ChangeSizeBitMap解析
/** * 等比例缩放图片 * * @param bt * bitmap图片对象 * @param resize * 状态,是否进行缩放 * @return bitmap图片对象 */ publicstatic Bitmap ChangeSizeBitMap(Bitmap bt, int resize) { intsrc_width = bt.getWidth(); intsrc_height = bt.getHeight(); floatwidth = (float)bt.getWidth(); floatheight = (float)bt.getHeight(); floatbmpWidth = (float) resize / width; floatbmpHeight = (float) resize / height; if(width >= height) { if(width > resize) { bmpWidth= (float) resize / height; bmpHeight= bmpWidth; } }else{ if(height > resize) { bmpHeight= (float) resize / height; bmpWidth= bmpHeight; } } if(bmpWidth > 1 || bmpHeight > 1) { returnbt; } Matrixmatrix = new Matrix(); matrix.postScale(bmpWidth,bmpHeight); BitmapresizeBmp = Bitmap.createBitmap(bt, 0, 0, src_width, src_height, matrix, true); returnresizeBmp; }这个方法里的东西就很简单了,不过计算的方式可以参考下,和下载图片时候的差不多。
转自:http://blog.csdn.net/zhangyue0503/article/details/7258347