Android图片二级缓存
想起刚開始写代码的时候,领导叫我写一个头像下载的方法,当时屁颠屁颠就写了一个图片下载的,每次都要去网络上请求,最后直接被pass掉了
当时的思路是这种
后来渐渐地就知道了有二级缓存这东西。
自己也阅读过非常多关于双缓存的文章。
APP开发到越后面。对性能的要求越高。那么双缓存的优势就逐渐体现出来了。
所谓图片双缓存。首先到执行内存中请求,再到sd卡请求,最后到网络请求,流程图例如以下
那我们从第一部開始解析
1.先看 内存缓存的代码
- public class MemoryCache implements ImageCache {
- private static final String TAG = MemoryCache.class.getSimpleName();
- private LruCache<String,Bitmap> mMemoryCache;
- public MemoryCache(){
- init();
- }
- private void init(){
- final int maxMemory = (int) (Runtime.getRuntime().maxMemory()/1024);
- final int cacheSize = maxMemory/4;
- mMemoryCache = new LruCache<String,Bitmap>(cacheSize){
- @Override
- protected int sizeOf(String key, Bitmap value) {
- return value.getRowBytes()*value.getHeight()/1024;
- }
- };
- }
- @Override
- public Bitmap get(String key) {
- Bitmap bitmap = mMemoryCache.get(key);
- if (bitmap!=null){
- Log.i(TAG,"File is exist in memory");
- }
- return mMemoryCache.get(key);
- }
- @Override
- public void put(String key, Bitmap bitmap) {
- if (get(key)==null) {
- mMemoryCache.put(key, bitmap);
- }
- }
- }
- private void init()
init()方法中对一些变量进行初始化,mMemoryCache用于在内存中缓存图片
- public Bitmap get(String key) {}
get()方法用于从内存中获得缓存
- public void put(String key, Bitmap bitmap) {}
put()方法将下载好的图片缓存到内存中,方便下次使用
2.再看sd卡缓存
- public class DiskCache implements ImageCache {
- private static final String TAG = DiskCache.class.getSimpleName();
- static String mPath ;
- public DiskCache(Context context){
- init(context);
- }
- private void init(Context context){
- // 获取图片缓存路径
- mPath = getDiskCachePath(context,"bitmap");
- File cacheDir = new File(mPath);
- if (!cacheDir.exists()) {
- cacheDir.mkdirs();
- }
- }
- @Override
- public Bitmap get(String key) {
- File file = new File(mPath+key);
- if (file.exists()){
- return BitmapFactory.decodeFile(mPath+key);
- }
- return null;
- }
- @Override
- public void put(String key, Bitmap bitmap) {
- FileOutputStream fileOutputStream = null;
- try {
- File file = new File(mPath+key);
- if (file.exists()){
- Log.i(TAG,"File is exist on disk");
- }
- fileOutputStream = new FileOutputStream(mPath+key);
- bitmap.compress(Bitmap.CompressFormat.PNG,100,fileOutputStream);
- } catch (FileNotFoundException e) {
- e.printStackTrace();
- }finally {
- CloseUtils.closeQuietly(fileOutputStream);
- }
- }
- /**
- * 依据传入的dir获得路径
- * @param context
- * @param dir
- * @return
- */
- public String getDiskCachePath(Context context, String dir) {
- String cachePath;
- if (Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState())
- || !Environment.isExternalStorageRemovable()) {
- cachePath = context.getExternalCacheDir().getPath();
- } else {
- cachePath = context.getCacheDir().getPath();
- }
- return cachePath + File.separator + dir;
- }
- }
相同
- private void init()
init()方法中对一些变量进行初始化,mMemoryCache用于在内存中缓存图片
- public Bitmap get(String key) {}
get()方法用于从内存中获得缓存
- public void put(String key, Bitmap bitmap) {}
put()方法将下载好的图片缓存到内存中,方便下次使用
接下来我们会在一个叫DoubleCache的类中对以上两种缓存方式进行管理
- public class DoubleCache implements ImageCache {
- private static final String TAG = DoubleCache.class.getSimpleName();
- private MemoryCache mMemoryCache = null;
- private DiskCache mDiskCache = null;
- public DoubleCache(Context context){
- mMemoryCache = new MemoryCache();
- mDiskCache = new DiskCache(context);
- }
- @Override
- public Bitmap get(String url) {
- String key = url2Key(url);
- Bitmap bitmap = mMemoryCache.get(key);
- if(bitmap==null){
- bitmap = mDiskCache.get(key);
- }else {
- }
- return bitmap;
- }
- @Override
- public void put(String url, Bitmap bitmap) {
- String key = url2Key(url);
- mMemoryCache.put(key,bitmap);
- mDiskCache.put(key,bitmap);
- }
- //url转key
- private String url2Key(String url){
- String key = MD5.hashKeyForDisk(url)+".jpg";
- return key;
- }
- }
我们在获取缓存的时候先从内存中获取。当内存中击中直接返回,当内存中没有击中,则訪问sd卡。
3.看到这里,小伙伴们一定急了,这仅仅有从缓存中和sd卡中取图片,并没有从网络获取,别急。立即就来
- public class ImageLoader {
- private static final String TAG = ImageLoader.class.getSimpleName();
- private static ImageLoader sInstance;
- private DoubleCache mDoubleCache = null;
- private ExecutorService mExecutorService = Executors.newFixedThreadPool(Runtime.getRuntime().availableProcessors());
- private ImageLoader(Context context) {
- mDoubleCache = new DoubleCache(context);
- }
- public static ImageLoader getInstance(Context context) {
- if (sInstance == null) {
- synchronized (ImageLoader.class) {
- sInstance = new ImageLoader(context);
- }
- }
- return sInstance;
- }
- public void displayImage(String url, ImageView imageView) {
- Bitmap bitmap = mDoubleCache.get(url);
- if (bitmap != null) {
- imageView.setImageBitmap(bitmap);
- mDoubleCache.put(url,bitmap);
- return;
- }
- submitLoadRequest(url, imageView);
- }
- private void submitLoadRequest(final String url, final ImageView imageView) {
- Log.i(TAG,"Download,url:"+url);
- imageView.setTag(url);
- mExecutorService.submit(new Runnable() {
- @Override
- public void run() {
- final Bitmap bitmap = downloadImage(url);
- if (imageView.getTag().equals(url)) {
- imageView.post(new Runnable() {
- @Override
- public void run() {
- imageView.setImageBitmap(bitmap);
- }
- });
- }
- mDoubleCache.put(url, bitmap);
- }
- });
- }
- Handler handler = new Handler(){
- @Override
- public void handleMessage(Message msg) {
- }
- };
- public Bitmap downloadImage(String url) {
- Bitmap bitmap = null;
- HttpURLConnection conn = null;
- try {
- URL url1 = new URL(url);
- conn = (HttpURLConnection) url1.openConnection();
- bitmap = BitmapFactory.decodeStream(conn.getInputStream());
- if (bitmap!=null){
- mDoubleCache.put(url,bitmap);
- }
- } catch (MalformedURLException e) {
- e.printStackTrace();
- } catch (Exception e) {
- e.printStackTrace();
- } finally {
- if (conn != null) {
- conn.disconnect();
- }
- }
- return bitmap;
- }
从
- displayImage()
方法中能够看到。假设缓存中都没有才从网络中获取
- public Bitmap downloadImage(String url) {}
下载完毕之后。把图片放到缓存中。
到这里,我们的第二张流程图就走完了。是不是非常easy。
我们在看一下是怎样使用的
- private ImageView imageView;
- private ImageView imageView2;
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_main);
- imageView = (ImageView) findViewById(R.id.image);
- imageView2 = (ImageView) findViewById(R.id.image2);
- imageView.setOnClickListener(new View.OnClickListener() {
- @Override
- public void onClick(View v) {
- ImageLoader.getInstance(MainActivity.this).displayImage("http://img.my.csdn.net/uploads/201407/26/1406383299_1976.jpg", imageView);
- }
- });
- imageView2.setOnClickListener(new View.OnClickListener() {
- @Override
- public void onClick(View v) {
- ImageLoader.getInstance(MainActivity.this).displayImage("http://img.my.csdn.net/uploads/201407/26/1406383299_1976.jpg", imageView2);
- }
- });
- }
Android图片二级缓存的更多相关文章
- picasso_强大的Android图片下载缓存库
tag: android pic skill date: 2016/07/09 title: picasso-强大的Android图片下载缓存库 [本文转载自:泡在网上的日子 参考:http://bl ...
- 毕加索的艺术——Picasso,一个强大的Android图片下载缓存库,OkHttpUtils的使用,二次封装PicassoUtils实现微信精选
毕加索的艺术--Picasso,一个强大的Android图片下载缓存库,OkHttpUtils的使用,二次封装PicassoUtils实现微信精选 官网: http://square.github.i ...
- 【第五篇】Volley代码修改之图片二级缓存以及相关源码阅读(重写ImageLoader.ImageCache)
前面http://www.cnblogs.com/androidsuperman/p/8a157b18ede85caa61ca5bc04bba43d0.html 有讲到使用LRU来处理缓存的,但是只是 ...
- picasso-强大的Android图片下载缓存库
编辑推荐:稀土掘金,这是一个针对技术开发者的一个应用,你可以在掘金上获取最新最优质的技术干货,不仅仅是Android知识.前端.后端以至于产品和设计都有涉猎,想成为全栈工程师的朋友不要错过! pica ...
- Android 图片三级缓存之内存缓存(告别软引用(SoftRefrerence)和弱引用(WeakReference))
因为之前项目同事使用了图片三级缓存,今天整理项目的时候发现同事还是使用了软引用(SoftRefrerence)和弱引用(WeakReference),来管理在内存中的缓存.看到这个我就感觉不对了.脑海 ...
- Android 图片三级缓存
图片缓存的原理 实现图片缓存也不难,需要有相应的cache策略.这里采用 内存-文件-网络 三层cache机制,其中内存缓存包括强引用缓存和软引用缓存(SoftReference),其实网络不算cac ...
- android图片的缓存--节约内存提高程序效率
如今android应用占内存一个比一个大,android程序的质量亟待提高. 这里简单说说网络图片的缓存,我这边就简单的说说思路 1:网络图片,无疑须要去下载图片,我们不须要每次都去下载. 维护一张表 ...
- 如何使用picasso 对Android图片下载缓存
相比较其他,picasso的图片缓存更加简单一些,他只需要一行代码就可以表述:导入相关jar包 Picasso.with(context).load("图片路径").into(Im ...
- Android图片载入缓存框架Glide
Glide开源框架是Google推荐的图片载入和缓框架,其在Github上的开源地址是:https://github.com/bumptech/glide 当然一个Google推荐的框架肯定就是Vol ...
随机推荐
- 关于热插拔usb hotplug /proc/sys/kernel mdev udev b...
转:http://www.360doc.com/content/10/0527/18/9922_29835045.shtml 这篇文章说的很好http://blog.chinaunix.net/u1/ ...
- nio selector
为什么使用Selector? 仅用单个线程来处理多个Channels的好处是,只需要更少的线程来处理通道.事实上,可以只用一个线程处理所有的通道.对于操作系统来说,线程之间上下文切换的开销很大,而且每 ...
- 【fastJSON】利用fastJSON处理循环引用的问题
下载fastJSON jar com.alibaba.fastjson 第一种:[写死的] 将需要序列化的字段传递进去,得到结果 //需要序列化的实体+字段 SimplePropertyPreFi ...
- [Android Pro] 关于BitmapFactory.decodeStream(is)方法无法正常解码为Bitmap对象的解决方法
在android sdk 1.6版本API帮助文档中,其中关于BitmapFactory.decodeFactory.decodeStream(InputStream is)的帮助文档是这么说明的: ...
- 自定义标签(JspFragment类、invoke方法、开发带属性的标签)
自定义标签(JspFragment类.invoke方法.开发带属性的标签) 一.JspFragment类 javax.servlet.jsp.tagext.JspFragment类是在JSP2.0中定 ...
- Tomcat集群环境下session共享方案 通过memcached 方法实现
对于web应用集群的技术实现而言,最大的难点就是:如何能在集群中的多个节点之间保持数据的一致性,会话(Session)信息是这些数据中最重要的一块.要实现这一点, 大体上有两种方式:一种是把所有Ses ...
- android简单的答题游戏
笔者最近开始沦陷于android,从开始入门到现在已经快半个月的时间,于是便写一个较综合,用到了数据库,多线程操作,以及时钟的添加和停止消除,activity之间的动画转换等,适用于初学者学以致用的小 ...
- JAVA NIO 之ByteBuffer的mark、position、limit、flip、reset,get方法介绍
参考博客:http://blog.csdn.net/sunzhenhua0608/article/details/31778519 先来一个demo: import java.nio.ByteBuff ...
- EXCEL在使用中,jar导入问题
报错:The type org.apache.poi.ss.usermodel.Workbook cannot be resolved. It is indirectly referenced fro ...
- vue - index.html
描述:通过App.vue绑定#App元素(进行展示).div#id中间写入任何内容都会失效. <!DOCTYPE html> <html> <head> <m ...