Android 图片的三级缓存机制实例分析

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);
}

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

(0)

相关推荐

  • Android图片三级缓存的原理及其实现

    为什么要使用三级缓存 如今的 Android App 经常会需要网络交互,通过网络获取图片是再正常不过的事了 假如每次启动的时候都从网络拉取图片的话,势必会消耗很多流量.在当前的状况下,对于非wifi用户来说,流量还是很贵的,一个很耗流量的应用,其用户数量级肯定要受到影响 特别是,当我们想要重复浏览一些图片时,如果每一次浏览都需要通过网络获取,流量的浪费可想而知 所以提出三级缓存策略,通过网络.本地.内存三级缓存图片,来减少不必要的网络交互,避免浪费流量 什么是三级缓存 网络缓存, 不优先加载,

  • Android中图片的三级缓存机制

    我们不能每次加载图片的时候都让用户从网络上下载,这样不仅浪费流量又会影响用户体验,所以Android中引入了图片的缓存这一操作机制. 原理: 首先根据图片的网络地址在网络上下载图片,将图片先缓存到内存缓存中,缓存到强引用中 也就是LruCache中.如果强引用中空间不足,就会将较早存储的图片对象驱逐到软引用(softReference)中存储,然后将图片缓存到文件(内部存储外部存储)中:读取图片的时候,先读取内存缓存,判断强引用中是否存在图片,如果强引用中存在,则直接读取,如果强引用中不存在,则

  • 详解Android中图片的三级缓存及实例

    详解Android中图片的三级缓存及实例 为什么要使用三级缓存 如今的 Android App 经常会需要网络交互,通过网络获取图片是再正常不过的事了 假如每次启动的时候都从网络拉取图片的话,势必会消耗很多流量.在当前的状况下,对于非wifi用户来说,流量还是很贵的,一个很耗流量的应用,其用户数量级肯定要受到影响 特别是,当我们想要重复浏览一些图片时,如果每一次浏览都需要通过网络获取,流量的浪费可想而知 所以提出三级缓存策略,通过网络.本地.内存三级缓存图片,来减少不必要的网络交互,避免浪费流量

  • android中图片的三级缓存cache策略(内存/文件/网络)

    1.简介 现在android应用中不可避免的要使用图片,有些图片是可以变化的,需要每次启动时从网络拉取,这种场景在有广告位的应用以及纯图片应用(比如百度美拍)中比较多. 现在有一个问题:假如每次启动的时候都从网络拉取图片的话,势必会消耗很多流量.在当前的状况下,对于非wifi用户来说,流量还是很贵的,一个很耗流量的应用,其用户数量级肯定要受到影响.当然,我想,向百度美拍这样的应用,必然也有其内部的图片缓存策略.总之,图片缓存是很重要而且是必须的. 2.图片缓存的原理 实现图片缓存也不难,需要有相

  • Android使用缓存机制实现文件下载及异步请求图片加三级缓存

    首先给大家介绍Android使用缓存机制实现文件下载 在下载文件或者在线浏览文件时,或者为了保证文件下载的正确性,需要使用缓存机制,常使用SoftReference来实现. SoftReference的特点是它的一个实例保存对一个Java对象的软引用,该软引用的存在不妨碍垃圾收集线程对该Java对象的回收.也就是说,一旦SoftReference保存了对一个Java对象的软引用后,在垃圾线程对这个Java对象回收前,SoftReference类所提供的get()方法返回Java对象的强引用.另外

  • 浅谈Android 中图片的三级缓存策略

    什么是三级缓存? 内存缓存,优先加载,速度最快 本地缓存,次优先加载,速度快 网络缓存,最后加载,速度慢,浪费流量 为什么要进行三级缓存 三级缓存策略,最实在的意义就是 减少不必要的流量消耗,增加加载速度 . 如今的 APP 网络交互似乎已经必不可少,通过网络获取图片再正常不过了.但是,每次启动应用都要从网络获取图片,或者是想重复浏览一些图片的时候,每次浏览都需要网络获取,消耗的流量就多了,在如今的流量资费来说,肯定会容易影响用户数量. 还有就是网络加载图片,有时候会加载很慢,影响了用户体验.

  • Android图片三级缓存策略(网络、本地、内存缓存)

    一.简介 现在的Android应用程序中,不可避免的都会使用到图片,如果每次加载图片的时候都要从网络重新拉取,这样不但很耗费用户的流量,而且图片加载的也会很慢,用户体验很不好.所以一个应用的图片缓存策略是很重要的.通常情况下,Android应用程序中图片的缓存策略采用"内存-本地-网络"三级缓存策略,首先应用程序访问网络拉取图片,分别将加载的图片保存在本地SD卡中和内存中,当程序再一次需要加载图片的时候,先判断内存中是否有缓存,有则直接从内存中拉取,否则查看本地SD卡中是否有缓存,SD

  • 详解Android 图片的三级缓存及图片压缩

    为什么需要图片缓存 Android默认给每个应用只分配16M的内存,所以如果加载过多的图片,为了防止内存溢出,应该将图片缓存起来.图片的三级缓存分别是: 内存缓存 本地缓存 网络缓存 其中,内存缓存应优先加载,它速度最快:本地缓存次优先加载,它速度也快:网络缓存不应该优先加载,它走网络,速度慢且耗流量. 三级缓存的具体实现 网络缓存 根据图片的url去加载图片 在本地和内存中缓存 public class NetCacheUtils { private LocalCacheUtils mLoca

  • Android实现图片异步请求加三级缓存

    使用xUtils等框架是很方便,但今天要用代码实现bitmapUtils 的功能,很简单, AsyncTask请求一张图片 ####AsyncTask #####AsyncTask是线程池+handler的封装 第一个泛型: 传参的参数类型类型(和doInBackground一致) 第二个泛型: #####更新进度的参数类型(和onProgressUpdate一致) 第三个泛型: 返回结果的参数类型(和onPostExecute一致, #####和doInBackground返回类型一致) 看A

  • Android 图片的三级缓存机制实例分析

    Android 图片的三级缓存机制实例分析 当我们获取图片的时候,如果不加以协调好图片的缓存,就会造成大流量,费流量应用,用户体验不好,影响后期发展.为此,我特地分享Android图片的三级缓存机制之从网络中获取图片,来优化应用,具体分三步进行: (1)从缓存中获取图片 (2)从本地的缓存目录中获取图片,并且获取到之后,放到缓存中 (3)从网络去下载图片,下载完成之后,保存到本地和放到缓存中 很好的协调这三层图片缓存就可以大幅度提升应用的性能和用户体验. 快速实现三级缓存的工具类ImageCac

  • 关于Android的 DiskLruCache磁盘缓存机制原理

    目录 一.为什么用DiskLruCache 1.LruCache和DiskLruCache 2.为何使用DiskLruCache 二.DiskLruCache使用 1.添加依赖 2.创建DiskLruCache对象 3.添加 / 获取 缓存(一对一) 4.添加 / 获取 缓存(一对多) 三.源码分析 1.open() 2.rebuildJournal() 3.readJournal() 4.get() 5.validateKey 6.trimTOSize() 7.journalRebuildRe

  • MyBatis 动态SQL和缓存机制实例详解

    有的时候需要根据要查询的参数动态的拼接SQL语句 常用标签: - if:字符判断 - choose[when...otherwise]:分支选择 - trim[where,set]:字符串截取,其中where标签封装查询条件,set标签封装修改条件 - foreach: if案例 1)在EmployeeMapper接口文件添加一个方法 public Student getStudent(Student student); 2)如果要写下列的SQL语句,只要是不为空,就作为查询条件,如下所示,这样

  • Android图片实现压缩处理的实例代码

    整理文档,搜刮出一个Android图片实现压缩处理的实例代码,稍微整理精简一下做下分享. 详解: 1.获取本地图片File文件 获取BitmapFactory.Options对象 计算原始图片 目标图片宽高比 计算输出的图片宽高 2.根据宽高比计算options.inSampleSize值(缩放比例 If set to a value > 1, requests the decoder to subsample the original image, returning a smaller i

  • Hibernate缓存机制实例代码解析

    本文研究的主要是Hibernate缓存机制的相关内容,具体如下. 演示项目: Student.java: public class Student { /*学生ID*/ private int id; /*学生姓名*/ private String name; /*学生和班级的关系*/ private Classes classes; //省略setter和getter方法 } Classes.java: public class Classes { /*班级ID*/ private int i

  • Spring缓存机制实例代码

    Spring的缓存机制非常灵活,可以对容器中任意Bean或者Bean的方法进行缓存,因此这种缓存机制可以在JavaEE应用的任何层次上进行缓存. Spring缓存底层也是需要借助其他缓存工具来实现,例如EhCache(Hibernate缓存工具),上层则以统一API编程. 要使用Spring缓存,需要以下三步 1.向Spring配置文件导入context:命名空间 2.在Spring配置文件启用缓存,具体是添加 <cache:annotation-driven cache-manager="

随机推荐