直接应用项目中的Android图片缓存技术

前不久搞的Android图片缓存,刚开始引入开源的框架,用着还行,但是在开发中遇到问题,就比如universal-image-loader-1.9.5.jar这个框架吧,在加载图片的时候自定义imageview无法加载,可能是存在以下问题吧,况且导入框架导致开发的项目包越来越大,基于上面的这几种情况,于是我就想自己写一个图片三级缓存的工具。
简要分析:刚开始想,图片的加载显示无非是先检查内存里面有没有,没就去文件里面找,若是文件里面没有的话就去开启网络下载,这样也符合开发中的大部分需求,而且效率肯定要高一点,在某些场合如快速滑动listview时从文件中加载图片就没有从内存中加载得快。先来看看效果吧!
这是第一次加载出来的图片,当然是网上的一张图片:

图片加载出来了,我们要看看日志,日志不会说谎,我截图的日志

刚才是本地和内存都没有那就的去网络下载了,所有日志打印的是开启网络下载
然而第二次进入应用的时候呢?还回去开启网络下载吗?那肯定不会的,要不这样要它有啥用呢?
看看吧这就是第二次进入应用的时候的日志

从文件中加载图片,不在需要开启网络下载直接从文件中获取,好了,这样问题就来了,从文件中读取肯定效率不好呀,没有直接从内存中读取快吧!我们调用系统维护好的一个算法,最近最少使用算法,这个算法是在我学操作系统的时候学到的,当时老师讲了一下,不过还好,能在项目中用到老师讲的东西,对此,以为就这个就完了?我们应该把文件中的图片读取到内存中,这样防止像listview这样的快速滑动产生的错位bug。下面看看我在快速滑动时的效果:

加载的图片都是从内存中获取,这样在效果上面就很好的啦!
好的,这么好的图片三级缓存,那到底怎么实现的呢以及到底怎么用的呢?下面就贴出我写的代码吧!让小伙伴看看,顺便优化优化!

package com.example.util;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.Bitmap.CompressFormat;
import android.graphics.BitmapFactory;
import android.os.Handler;
import android.os.Message;
import android.support.v4.util.LruCache;
import android.util.Log;

/**
 * 图片的三级缓存工具类{日后项目需要}
 * @author double 江
 *
 */
public class ImageCachceUitl {
  public static final int SUCCSEE = 0;
  public static final int FAIL = 1;
  private Context context;
  private LruCache<String, Bitmap> cache;//lru算法集合,string是图片的url,bitmap为图片的值类型
  private File cacheDir;
  Handler handler;
  private ExecutorService executorService;//维护线性池
  public ImageCachceUitl(Context context ,Handler handler){
    this.context=context;
    this.handler=handler;
    cacheDir=context.getCacheDir();//获得cache文件夹
    //维护几个网络线程下载图片
    executorService=Executors.newFixedThreadPool(5);
    int maxSize=(int) (Runtime.getRuntime().maxMemory()/8);//获得运行环境的内存大小的1/8
    cache=new LruCache<String, Bitmap>(maxSize){
      // TODO 每存储一张图片的大小(作用于内存溢出丢图片)
      @Override
      protected int sizeOf(String key, Bitmap value) {
        //返回当前一行所占的字节数*高度,就是图片的大小
        return value.getRowBytes()*value.getHeight();
      }
    };//当前图片缓存总数的大小
  }
  /**
   *
   * @param url 下载图片的连接
   * @param position 需要显示图片的imgeView的Tag
   * @return
   */
  public Bitmap getBitmapFromUrl(String url,int position){
    //1内存中获取图片LRU算法
    Bitmap bitmap=cache.get(url);
    //内存中有指定图片
    if (bitmap!=null) {
      Log.i("从内存中获得图片", "从内存中获得图片"+url);
      return bitmap;
    }
    //2文件中获取图片
    bitmap=getBitmapFromFile(url);
    if (bitmap!=null) {
      Log.i("从文件中获得图片", "从文件中获得图片"+url);
      return bitmap;
    }
    //3开启网络下载
    Log.i("从网络中获得图片", "从网络中获得图片"+url);
    getBitmapFromNet(url,position);
    return null;

  }

  /**
   * 网络获取图片
   * @param url 图片的;链接地址
   * @param position 需要显示的imageview的tag
   */
  private void getBitmapFromNet(String url, int position) {

    //开启任务
    executorService.execute(new RunnableTask(url,position));
  }
  //任务接口
  class RunnableTask implements Runnable{
    String imageUrl;
    int position;
    private HttpURLConnection httpURLConnection;
    public RunnableTask(String imageUrl, int position) {
      this.position=position;
      this.imageUrl=imageUrl;
    }

    @Override
    public void run() {
      //子线程的操作,开启网络下载图片
      try {
        URL url=new URL(imageUrl);
        //请求对象
         httpURLConnection=(HttpURLConnection) url.openConnection();
        //网络请求的方式
        httpURLConnection.setRequestMethod("GET");
        //超时的时间,
        httpURLConnection.setConnectTimeout(5000);
        //读取超时的时间
        httpURLConnection.setReadTimeout(5000);
        //httpURLConnection.getResponseCode()拿到最新数据
        if (httpURLConnection.getResponseCode()==200) {
          //success get data from net;get tape
          InputStream inputStream=httpURLConnection.getInputStream();
          //将流转化成bitmap图片
          Bitmap bitmap=BitmapFactory.decodeStream(inputStream);
          //利用handler机制放入主线程中显示
          Message msg=new Message();
          //需要在主线程中显示的图片msg.obj
          msg.obj=bitmap;
          msg.arg1=position;
          //为msg设置标记
          msg.what=SUCCSEE;
          handler.sendMessage(msg);
          //一,将下载完后的图片保存到内存中
          cache.put(imageUrl, bitmap);
          //二,将下载完后的图片保存到文件中
          writeToLoce(imageUrl,bitmap);
          return;
        }

      } catch (Exception e) {
        e.printStackTrace();
      }
      //关闭请求
      finally{
        //断开服务器
        if (httpURLConnection!=null) {
          httpURLConnection.disconnect();
        }
      }
      //发送一个空消息
      handler.obtainMessage(FAIL).sendToTarget();
    }
  }
  /**
   * 图片写入cache文件夹下面的操作
   * @param imageUrl
   * @param bitmap
   */
  private void writeToLoce(String imageUrl, Bitmap bitmap) {
    try {
      String bitmapefilename=MD5Encoder.encode(imageUrl).substring(10);
      Log.i("bitmapefilename", bitmapefilename);
      File file=new File(cacheDir, bitmapefilename);
      FileOutputStream fileOutputStream =new FileOutputStream(file);
      //写入文件的操作(1图片类型2图片质量当为100时表示不压缩3文件流)
      bitmap.compress(CompressFormat.JPEG, 100, fileOutputStream);
    } catch (Exception e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    }
  }
  /**
   * 读取文件中图片的操作
   * @param url 图片的连接地址
   * @return
   */
  private Bitmap getBitmapFromFile(String url) {
    try {
      //使用Md5工具加密截取前10个字符串
      String bitmapefilename=MD5Encoder.encode(url).substring(10);
      /**
       * 在cache文件夹下面找到指定文件
       * dir cache文件存储路径
       * name 文件名称
       */
      File file=new File(cacheDir, bitmapefilename);
    // file.mkdir();
      //FileInputStream fileInputStream=new FileInputStream(file);
      Bitmap bitmap=BitmapFactory.decodeFile(file.getPath());//完整文件路径
      //Log.i("文件路径", file.getPath().toString());
      //2读取之后放入内存,提高效率
      cache.put(url, bitmap);
      return bitmap;
    } catch (Exception e) {
      e.printStackTrace();
    }
    return null;

  }
}

下面就是一个使用demo了,其实很简单


package com.example.do0me;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.security.PublicKey;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

import org.ksoap2.SoapEnvelope;
import org.ksoap2.serialization.MarshalBase64;
import org.ksoap2.serialization.SoapObject;
import org.ksoap2.serialization.SoapSerializationEnvelope;
import org.ksoap2.transport.HttpTransportSE;
import org.xmlpull.v1.XmlPullParserException;

import android.annotation.SuppressLint;
import android.app.Activity;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.drawable.BitmapDrawable;
import android.os.Bundle;
import android.os.Handler;
import android.util.Base64;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
import android.view.ViewTreeObserver.OnPreDrawListener;
import android.widget.BaseAdapter;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.Toast;

import com.example.util.AgbcApi;
import com.example.util.ClippingPicture;
import com.example.util.FastBlur;
import com.example.util.ImageCachceUitl;
import com.tencent.connect.avatar.c;

/**
 * 图片三级缓存的测试+图片上传
 * @author double 江
 *
 */
public class MainActivity4 extends Activity {
  private ImageCachceUitl imageCachceUitl;
  private List<String> urlList=new ArrayList<String>();
  private Runnable runnable;
  private Handler handler =new Handler(){
    public void handleMessage(android.os.Message msg) {
      switch (msg.what) {
      case ImageCachceUitl.SUCCSEE:
        Bitmap bitmap=(Bitmap) msg.obj;
        int psition=msg.arg1;
        //通过TAg加载当前的limageview
        ImageView imageView=(ImageView) listview.findViewWithTag(psition);
        if (null!=bitmap&&null!=imageView) {
          imageView.setImageBitmap(bitmap);
        }
        break;
      case ImageCachceUitl.FAIL:
        Toast.makeText(getApplicationContext(), "下载错误", Toast.LENGTH_LONG).show();
      default:
        break;
      }
    };
  };
  private ListView listview;
  private SoapObject request;
  private ExecutorService executorService;
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main4);
    listview=(ListView) findViewById(R.id.imageviewlist);
    imageCachceUitl=new ImageCachceUitl(getApplicationContext(), handler);
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");

    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");
    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");

    urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");urlList.add("http://ww4.sinaimg.cn/large/90bd89ffjw1eqvmd6o8r6j20go0p5ju2.jpg");

    listview.setAdapter(new myListAdapt());
    executorService=Executors.newFixedThreadPool(5);
    executorService.execute(new Runnable() {

      @Override
      public void run() {
        getImageromSdk();
      }
    });
  }

  public String getImageFromAndroid(String arg0,String arg1, String arg2){
    Log.i("进入端口方法", "进入端口方法");
    final String methodName="getImageFromAndroid";
    final String soapAction=AgbcApi.NAMESPACE+methodName;
    request = new SoapObject(AgbcApi.NAMESPACE, methodName);
    request.addProperty("arg0",arg1);
    request.addProperty("arg1",arg2);
    SoapSerializationEnvelope envelope = new SoapSerializationEnvelope(SoapEnvelope.VER11);
    (new MarshalBase64()).register(envelope);
    envelope.bodyOut = request;
    envelope.dotNet=false;
    envelope.setOutputSoapObject(request);
    HttpTransportSE ht = new HttpTransportSE(AgbcApi.TASKSERVICEURL);
    ht.debug=true;
    try {

      ht.call(soapAction, envelope);
      Log.i("请求", envelope.bodyIn.toString());
    } catch (IOException | XmlPullParserException e) {
      e.printStackTrace();
    }
    return arg1;

  };

  @SuppressLint("SdCardPath")
  public void getImageromSdk(){
    Log.i("进入获取图片方法", "进入获取图片方法");
    try{
      String srcUrl = "/sdcard/"; //路径
      String fileName = "1.png"; //文件名
      String filrName2="2.jpg";
    List<String>imageList=new ArrayList<>();
    imageList.add(fileName);
    imageList.add(filrName2);
    for (int i = 0; i < imageList.size(); i++) {
      FileInputStream fis = new FileInputStream(srcUrl + imageList.get(i));
      ByteArrayOutputStream baos = new ByteArrayOutputStream();
      byte[] buffer = new byte[4096];
      int count = 0;
      while((count = fis.read(buffer)) >= 0){
        baos.write(buffer, 0, count);
      }
      String uploadBuffer = new String(Base64.encode(baos.toByteArray(), Base64.DEFAULT)); //进行Base64编码
      String methodName = "uploadImage";
        getImageFromAndroid(methodName,imageList.get(i), uploadBuffer);  //调用webservice
      Log.i("connectWebService", "start");
      fis.close();
}
    }catch(Exception e){
      e.printStackTrace();
    }
  }
  class myListAdapt extends BaseAdapter{
    private LayoutInflater layoutInflater;
    ImageView list_imag;
    Button list_but;
    @Override
    public int getCount() {
      // TODO Auto-generated method stub
      return urlList.size();
    }

    @Override
    public Object getItem(int position) {
      // TODO Auto-generated method stub
      return urlList.get(position);
    }

    @Override
    public long getItemId(int position) {
      // TODO Auto-generated method stub
      return position;
    }

    @SuppressLint({ "InflateParams", "ViewHolder" })
    @Override
    public View getView(final int position, View convertView, ViewGroup parent) {

      layoutInflater=LayoutInflater.from(getApplication());
      //if (layoutInflater==null) {
      convertView = layoutInflater.inflate(R.layout.image_list_item, null);
      // }
      list_but=(Button) convertView.findViewById(R.id.list_but);
      list_imag=(ImageView) convertView.findViewById(R.id.list_imag);

      list_imag.setTag(position);
      final Bitmap bitmap=imageCachceUitl.getBitmapFromUrl(urlList.get(position), position);
      if (null!=bitmap) {
        list_imag.setImageBitmap(bitmap);
      }
      list_imag.setVisibility(View.VISIBLE);

      list_but.setOnClickListener(new OnClickListener() {

        @Override
        public void onClick(View v) {
          //urlList.get(position);
          Log.i("获取点击焦点", "获取点击焦点");
          // getImageromSdk();
          //startRun();
          new Thread(new Runnable() {

            @Override
            public void run() {
              getImageromSdk();
            }
          }).start();
        }
      });
      return convertView;
    }

  }

}

就为大家介绍到这吧,大家慢慢消化,希望文章对大家有或多或少的帮助。

(0)

相关推荐

  • Android图片缓存之初识Glide(三)

    前言: 前面总结学习了图片的使用以及Lru算法,今天来学习一下比较优秀的图片缓存开源框架.技术本身就要不断的更迭,从最初的自己使用SoftReference实现自己的图片缓存,到后来做电商项目自己的实现方案不能满足项目的需求改用Afinal,由于Afinal不再维护而选择了师出同门的Xutils,中间也接触过别的开源框架比如Picasso,对Picasso的第一次印象就不太好,初次接触是拿到了公司刚从外包公司接手过来的图片社交类app,对内存占用太大,直接感受就是导致ListView滑动有那么一

  • Android图片缓存之Bitmap详解(一)

    前言: 最近准备研究一下图片缓存框架,基于这个想法觉得还是先了解有关图片缓存的基础知识,今天重点学习一下Bitmap.BitmapFactory这两个类.  Bitmap: Bitmap是Android系统中的图像处理的最重要类之一.用它可以获取图像文件信息,进行图像剪切.旋转.缩放等操作,并可以指定格式保存图像文件.  重要函数  •public void recycle() // 回收位图占用的内存空间,把位图标记为Dead  •public final boolean isRecycled

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

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

  • Android中Glide加载库的图片缓存配置究极指南

    零.选择Glide 为什么图片加载我首先推荐Glide? 图片加载框架用了不少,从afinal框架的afinalBitmap,Xutils的BitmapUtils,老牌框架universalImageLoader,著名开源组织square的picasso,google推荐的glide到FaceBook推出的fresco.这些我前前后后都体验过,那么面对这么多的框架,该如何选择呢?下面简单分析下我的看法. afinal和Xuils在github上作者已经停止维护了,开源社区最新的框架要属KJFra

  • Android实现图片缓存与异步加载

    ImageManager2这个类具有异步从网络下载图片,从sd读取本地图片,内存缓存,硬盘缓存,图片使用动画渐现等功能,已经将其应用在包含大量图片的应用中一年多,没有出现oom. Android程序常常会内存溢出,网上也有很多解决方案,如软引用,手动调用recycle等等.但经过我们实践发现这些方案,都没能起到很好的效果,我们的应用依然会出现很多oom,尤其我们的应用包含大量的图片.android3.0之后软引用基本已经失效,因为虚拟机只要碰到软引用就回收,所以带不来任何性能的提升. 我这里的解

  • android上的一个网络接口和图片缓存框架enif简析

    1.底层网络接口采用apache的httpclient连接池框架: 2.图片缓存采用基于LRU的算法: 3.网络接口采用监听者模式: 4.包含图片的OOM处理(及时回收处理技术的应用): 图片核心处理类:CacheView.java 复制代码 代码如下: package xiaogang.enif.image; import java.io.FilterInputStream; import java.io.IOException; import java.io.InputStream; imp

  • android异步加载图片并缓存到本地实现方法

    在android项目中访问网络图片是非常普遍性的事情,如果我们每次请求都要访问网络来获取图片,会非常耗费流量,而且图片占用内存空间也比较大,图片过多且不释放的话很容易造成内存溢出.针对上面遇到的两个问题,首先耗费流量我们可以将图片第一次加载上面缓存到本地,以后如果本地有就直接从本地加载.图片过多造成内存溢出,这个是最不容易解决的,要想一些好的缓存策略,比如大图片使用LRU缓存策略或懒加载缓存策略.今天首先介绍一下本地缓存图片. 首先看一下异步加载缓存本地代码: 复制代码 代码如下: public

  • Android图片缓存之Lru算法(二)

    前言: 上篇我们总结了Bitmap的处理,同时对比了各种处理的效率以及对内存占用大小,点击查看.我们得知一个应用如果使用大量图片就会导致OOM(out of memory),那该如何处理才能近可能的降低oom发生的概率呢?之前我们一直在使用SoftReference软引用,SoftReference是一种现在已经不再推荐使用的方式,因为从 Android 2.3 (API Level 9)开始,垃圾回收器会更倾向于回收持有软引用或弱引用的对象,这让软引用变得不再可靠,所以今天我们来认识一种新的缓

  • Android开发笔记之图片缓存、手势及OOM分析

    把图片缓存.手势及OOM三个主题放在一起,是因为在Android应用开发过程中,这三个问题经常是联系在一起的.首先,预览大图需要支持手势缩放,旋转,平移等操作:其次,图片在本地需要进行缓存,避免频繁访问网络:最后,图片(Bitmap)是Android中占用内存的大户,涉及高清大图等处理时,内存占用非常大,稍不谨慎,系统就会报OOM错误. 庆幸的是,这三个主题在Android开发中属于比较普遍的问题,有很多针对于此的通用的开源解决方案.因此,本文主要说明笔者在开发过程中用到的一些第三方开源库.主要

  • Android图片缓存原理、特性对比

    这是我在 MDCC 上分享的内容(略微改动),也是源码解析第一期发布时介绍的源码解析后续会慢慢做的事. 从总体设计和原理上对几个图片缓存进行对比,没用到他们的朋友也可以了解他们在某些特性上的实现. 一. 四大图片缓存基本信息 Universal ImageLoader 是很早开源的图片缓存,在早期被很多应用使用. Picasso 是 Square 开源的项目,且他的主导者是 JakeWharton,所以广为人知. Glide 是 Google 员工的开源项目,被一些 Google App 使用,

随机推荐