Android 图片的三级缓存机制实例分析
当我们获取图片的时候,如果不加以协调好图片的缓存,就会造成大流量,费流量应用,用户体验不好,影响后期发展。为此,我特地分享Android图片的三级缓存机制之从网络中获取图片,来优化应用,具体分三步进行:
(1)从缓存中获取图片
(2)从本地的缓存目录中获取图片,并且获取到之后,放到缓存中
(3)从网络去下载图片,下载完成之后,保存到本地和放到缓存中
很好的协调这三层图片缓存就可以大幅度提升应用的性能和用户体验。
快速实现三级缓存的工具类ImageCacheUtil如下(有更好的建议可以发我的邮箱说出你的想法,一起完善,邮箱见博客主页“给我写信”):
1.从网络中获取图片的三级缓存工具类ImageCacheUtil
public class ImageCacheUtil { private LruCache<String, Bitmap> lruCache; private File cacheDir; private ExecutorService newFixedThreadPool; private Handler handler; public static final int SUCCESS = 100; public static final int FAIL = 101; //当我们获取图片的时候,分三步 //1.从缓存中获取图片 //2.从本地的缓存目录中获取图片,并且获取到之后,放到缓存中 //3.从网络去下载图片,下载完成之后,保存到本地缓存目录和放到缓存中 public ImageCacheUtil(Context context,Handler handler){ //获取缓存的大小 int maxsize = (int) (Runtime.getRuntime().maxMemory()/8); //maxSize : 设置缓存的最大空间 lruCache = new LruCache<String, Bitmap>(maxsize){ //获取移出的图片所占用的空间,当图片移出的时候,需要将图片占用的缓存空间从缓存中移出 @Override protected int sizeOf(String key, Bitmap value) { //计算图片所占用的缓存大小 //getRowBytes : 获取图片一行所占用的大小 //getHeight : 获取图片所占用行数 return value.getRowBytes()*value.getHeight(); } }; //获取缓存目录 cacheDir = context.getCacheDir(); //获取线程池 //nThreads : 线程池中的线程数量 newFixedThreadPool = Executors.newFixedThreadPool(5); this.handler = handler; } /** * 获取图片的方法 * @param url * @param positon * @return */ public Bitmap getBitmap(String url,int position){ Bitmap bitmap = null; //1.从缓存中获取图片 (LRUCache<k,v>) k:key 保存图片的标示,一般都是图片的url地址,v:value 图片 bitmap = lruCache.get(url);//根据key从缓存中获取相应的图片 //lruCache.put(url, bitmap):保存图片到缓存中 if (bitmap!=null) { return bitmap; } //2.从本地的缓存目录中获取图片,并且获取到之后,放到缓存中 bitmap = getFromLocal(url); if (bitmap!=null) { return bitmap; } //3.从网络去下载图片,下载完成之后,保存到本地缓存目录和放到缓存中 getFromNet(url,position); return null; } /** * 从网络下载图片,异步方式,线程池 * @param url * @param position */ private void getFromNet(String url, int position) { newFixedThreadPool.execute(new RunnableTask(url,position)); } class RunnableTask implements Runnable{ private String imageUrl; private int position; public RunnableTask(String url,int position){ this.imageUrl = url; this.position = position; } @Override public void run() { Message message = Message.obtain(); //下载图片 try { URL url = new URL(imageUrl); HttpURLConnection conn = (HttpURLConnection) url.openConnection(); conn.setReadTimeout(3000); conn.setConnectTimeout(5000); conn.setRequestMethod("GET"); InputStream inputStream = conn.getInputStream(); Bitmap bitmap = BitmapFactory.decodeStream(inputStream); //保存到本地缓存中 wirteToLocal(imageUrl, bitmap); //保存到系统缓冲中 lruCache.put(imageUrl, bitmap); //显示图片,给handler发送消息 message.what = SUCCESS; message.obj = bitmap; message.arg1 = position; handler.sendMessage(message); return; } catch (Exception e) { e.printStackTrace(); } message.what = FAIL; handler.sendMessage(message); } } /** * 从本地缓存目录获取图片 * @param url */ private Bitmap getFromLocal(String url) { //根据图片的名称获取图片 try { String fileName = MD5Encoder.encode(url).substring(10); File file = new File(cacheDir, fileName); Bitmap bitmap = BitmapFactory.decodeFile(file.getAbsolutePath()); //防盗缓存当中 lruCache.put(url, bitmap); return bitmap; } catch (Exception e) { e.printStackTrace(); } return null; } /** * 将图片保存到本地缓存目录中 */ public void wirteToLocal(String url,Bitmap bitmap){ //url名称,通过MD5加密,并且截取前10位作为名称 try { String fileName = MD5Encoder.encode(url).substring(10); File file = new File(cacheDir, fileName); FileOutputStream out = new FileOutputStream(file); //format :图片的格式(android中用的png多,因为png质量是不会改变的) //quality : 压缩比例 //stream : 流信息 bitmap.compress(CompressFormat.JPEG, 100, out);//将图片保存到那个位置 } catch (Exception e) { e.printStackTrace(); } } }
其中用到的MD5Encoder类如下:
public class MD5Encoder { public static String encode(String string) throws Exception { byte[] hash = MessageDigest.getInstance("MD5").digest(string.getBytes("UTF-8")); StringBuilder hex = new StringBuilder(hash.length * 2); for (byte b : hash) { if ((b & 0xFF) < 0x10) { hex.append("0"); } hex.append(Integer.toHexString(b & 0xFF)); } return hex.toString(); } }
2.在MainActivity初始化imageCacheUtil
ImageCacheUtil imageCacheUtil = new ImageCacheUtil(getApplicationContext, handler);
3.在MainActivity中通过handler将图片显示出来
图片通过工具类下载成功之后,不仅要将图片保存到本地缓存中和系统缓存中,还要将图片显示出来,通过handler进行处理,这个handler是设置使用ImageCacheUtil工具类,就要把你的handler传递过来,方便我们传消息给相应使用ImageCacheUtil工具类的类进行处理。
private Handler handler = new Handler(){ public void handleMessage(android.os.Message msg) { switch (msg.what) { case ImageCacheUtil.SUCCESS: //给控件设置图片 Bitmap bitmap = (Bitmap) msg.obj; int position = msg.arg1; ImageView image= (ImageView) view.findViewWithTag(position);//就是根据条目的位置获取相应的控件 if (image != null && bitmap != null) { image.setImageBitmap(bitmap); } break; case ImageCacheUtil.FAIL: Toast.makeText(getApplicationContext, "图片下载失败", 0).show(); break; } }; };
4.在MainActivity中的adapter的getview中进行调用
Bitmap bitmap = imageCacheUtil.getBitmap(list.get(position).listimage, position); if (bitmap != null) { viewHodler.image.setImageBitmap(bitmap); }
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!
本文向大家介绍Android中图片的三级缓存机制,包括了Android中图片的三级缓存机制的使用技巧和注意事项,需要的朋友参考一下 我们不能每次加载图片的时候都让用户从网络上下载,这样不仅浪费流量又会影响用户体验,所以Android中引入了图片的缓存这一操作机制。 原理: 首先根据图片的网络地址在网络上下载图片,将图片先缓存到内存缓存中,缓存到强引用中 也就是LruCache中。如果强引用中空
本文向大家介绍详解Android中图片的三级缓存及实例,包括了详解Android中图片的三级缓存及实例的使用技巧和注意事项,需要的朋友参考一下 详解Android中图片的三级缓存及实例 为什么要使用三级缓存 如今的 Android App 经常会需要网络交互,通过网络获取图片是再正常不过的事了 假如每次启动的时候都从网络拉取图片的话,势必会消耗很多流量。在当前的状况下,对于非wifi用户来说,流量
本文向大家介绍Android图片三级缓存开发,包括了Android图片三级缓存开发的使用技巧和注意事项,需要的朋友参考一下 因为目前工程无法使用第三方,只能搞一个三级缓存了三级缓存分为内存缓存,本地缓存,网络缓存;缓存的步骤依次是网络,内存,本地,然后取的顺序为内存,本地,网络。在加载图片时引用时尽量采用弱引用避免出现图片过多产生OOM.。 1、内存缓存,android为我们提供LruCache=
本文向大家介绍Android实现图片异步请求加三级缓存,包括了Android实现图片异步请求加三级缓存的使用技巧和注意事项,需要的朋友参考一下 使用xUtils等框架是很方便,但今天要用代码实现bitmapUtils 的功能,很简单, AsyncTask请求一张图片 ####AsyncTask #####AsyncTask是线程池+handler的封装 第一个泛型: 传参的参数类型类型(和doIn
本文向大家介绍Android图片三级缓存策略(网络、本地、内存缓存),包括了Android图片三级缓存策略(网络、本地、内存缓存)的使用技巧和注意事项,需要的朋友参考一下 一、简介 现在的Android应用程序中,不可避免的都会使用到图片,如果每次加载图片的时候都要从网络重新拉取,这样不但很耗费用户的流量,而且图片加载的也会很慢,用户体验很不好。所以一个应用的图片缓存策略是很重要的。通常情况下,A
本文向大家介绍Android 图片缓存机制的深入理解,包括了Android 图片缓存机制的深入理解的使用技巧和注意事项,需要的朋友参考一下 Android 图片缓存机制的深入理解 Android加载一张图片到用户界面是很简单的,但是当一次加载多张图片时,情况就变得复杂起来。很多情况下(像ListView、GridView或ViewPager等组件),屏幕上已显示的图片和即将滑动到当前屏幕上的图片数