当前位置: 首页 > news >正文

ImageLoader的优化写法

ImageLoader是android开发中常用到的图片缓存技术。本文通过阅读《Android源码设计模式解析与实战》后,了解到一个程序的健壮性,可拓展性是十分重要的。源码设计均来自于《Android源码设计模式解析与实战》: 1.首先定义ImageCache接口,实现图片的缓存和提取两种方法。

public interface ImageCache {
    void put(String url,Bitmap bitmap);
    Bitmap get(String url);

}
复制代码

2.分别实现内存缓存(MemoryCache),sd卡缓存(DiskCache)和双缓存(DoubleCache)策略,其均实现ImageCache接口。

public class MemoryCache implements ImageCache {
    LruCache<String, Bitmap> mMemoryCache;
    public MemoryCache(){
        initMemoryCache();
    }

    private void initMemoryCache() {
        //计算可使用的最大内存
        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) {
                //图片的宽高之积/1024(存储的应该是图片的大小)
                return value.getRowBytes() * value.getHeight() / 1024;
            }
        };
    }

    @Override
    public void put(String url, Bitmap bitmap) {
        mMemoryCache.put(url,bitmap);
    }

    @Override
    public Bitmap get(String url) {
        return mMemoryCache.get(url);
    }
}
复制代码
public class DiskCache implements ImageCache{
    static String cacheDir = "sdcard/cache/";

    @Override
    public void put(String url, Bitmap bitmap) {
        FileOutputStream fileOutputStream = null;
        try {
            fileOutputStream = new FileOutputStream(cacheDir + url);
            bitmap.compress(Bitmap.CompressFormat.PNG,100,fileOutputStream);
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        }finally {
            if(fileOutputStream != null){
                try {
                    fileOutputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    @Override
    public Bitmap get(String url) {
        return BitmapFactory.decodeFile(cacheDir + url);
    }
}
复制代码
public class DoubleCache implements ImageCache{
    ImageCache mMemmoryCache = new MemoryCache();
    ImageCache mDiskCache = new DiskCache();
    @Override
    public void put(String url, Bitmap bitmap) {
        mMemmoryCache.put(url,bitmap);
        mDiskCache.put(url,bitmap);
    }

    @Override
    public Bitmap get(String url) {
        Bitmap bitmap = mMemmoryCache.get(url);
        if(bitmap == null){
            bitmap = mDiskCache.get(url);
        }
        return bitmap;
    }
}
复制代码

3.创建ImageLoader类

public class ImageLoader {
    //内存缓存
    ImageCache mImageCache = new MemoryCache();
    //线程池,线程数量为CPU的数量
    ExecutorService mExecutorService = Executors.newFixedThreadPool(Runtime.getRuntime().availableProcessors());

    public void setImageCache(ImageCache cache){
        mImageCache = cache;
    }
    public void displayImage(final String url, final ImageView imageView) {
        //判断该url的图片是否在缓存中存在,存在则直接显示在imageview上
        Bitmap bitmap = mImageCache.get(url);
        if(bitmap != null){
            imageView.setImageBitmap(bitmap);
            return;
        }
        //图片没缓存,在线程池中下载
        submitLoadRequest(url,imageView);
    }

    private void submitLoadRequest(final String url, final ImageView imageView) {
        imageView.setTag(url);
        mExecutorService.submit(new Runnable() {
            @Override
            public void run() {
                Bitmap bitmap = downloadImage(url);
                if (bitmap == null) {
                    return;
                }
                if (imageView.getTag().equals(url)) {
                    imageView.setImageBitmap(bitmap);
                }
                mImageCache.put(url, bitmap);
            }
        });
    }

    private Bitmap downloadImage(String imageurl) {
        Bitmap bitmap = null;
        try {
            URL url = new URL(imageurl);
            HttpURLConnection conn = (HttpURLConnection) url.openConnection();
            bitmap = BitmapFactory.decodeStream(conn.getInputStream());
            conn.disconnect();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return bitmap;
    }
复制代码

在ImageLoader类中添加setImageCache()方法,用户可以通过此方法决定缓存的方式为以上3种的哪一种。

4.在Activity上进行声明,使用。

public class MainActivity extends AppCompatActivity {
    private ImageView imageView;
    private ImageLoader imageLoader;
    private String url = "http://bmob-cdn-11151.b0.upaiyun.com/2017/06/04/3cdfbcc8400de13c8027b5513d469637.jpg";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        initView();
    }

    private void initView() {
        imageView = (ImageView)findViewById(R.id.imageView);
        imageLoader = new ImageLoader();
        //使用内存缓存
        imageLoader.setImageCache(new MemoryCache());
        //使用sd卡缓存
        imageLoader.setImageCache(new DiskCache());
        //使用doubleCache缓存
        imageLoader.setImageCache(new DoubleCache());
        //使用自定义的方法缓存
        imageLoader.setImageCache(new ImageCache() {
            @Override
            public void put(String url, Bitmap bitmap) {
            }
            @Override
            public Bitmap get(String url) {
                return null;
            }
        });
        imageLoader.displayImage(url,imageView);
    }
}
复制代码

本方法高度内聚,低耦合,在使用过程中,分别使用3中不同的缓存策略,使复杂的代码进行分离,在出现问题时,便于方便的修改,不会影响到其他的代码功能,实现抽象接口的方法还允许开发者使用自定义的缓存策略,可拓展性高。

理解这种程序设计时十分必要的,在以后的开发中要时刻考虑自己程序的健壮性,可拓展性,高内聚低耦合。

相关文章:

  • 七牛实时音视频云视频连线demo(web部分)
  • 支付宝支撑2135亿成交额的数据库架构原理
  • ZStack--工作流引擎
  • dmidecode查看linux硬件信息
  • Octave 入门
  • 第33讲 | 区块链与供应链(二)
  • Hadoop完全分布式环境搭建(四)——基于Ubuntu16.04安装和配置Hadoop大数据环境...
  • 使用parted解决大于2T的磁盘分区
  • day29:关闭服务|
  • Python 爬虫十六式 - 第六式:JQuery的假兄弟-pyquery
  • 本科理工男如何学习Linux
  • [洛谷P2511][HAOI2008]木棍分割
  • C语言之路-2-判断
  • JavaScript面向对象名词详解
  • java对象拷贝最完全解说
  • 自己简单写的 事件订阅机制
  • 2017年终总结、随想
  • 2018以太坊智能合约编程语言solidity的最佳IDEs
  • Apache Zeppelin在Apache Trafodion上的可视化
  • C++回声服务器_9-epoll边缘触发模式版本服务器
  • Codepen 每日精选(2018-3-25)
  • CoolViewPager:即刻刷新,自定义边缘效果颜色,双向自动循环,内置垂直切换效果,想要的都在这里...
  • JS创建对象模式及其对象原型链探究(一):Object模式
  • js写一个简单的选项卡
  • Meteor的表单提交:Form
  • php的插入排序,通过双层for循环
  • use Google search engine
  • ViewService——一种保证客户端与服务端同步的方法
  • 案例分享〡三拾众筹持续交付开发流程支撑创新业务
  • 编写高质量JavaScript代码之并发
  • 不用申请服务号就可以开发微信支付/支付宝/QQ钱包支付!附:直接可用的代码+demo...
  • 工作踩坑系列——https访问遇到“已阻止载入混合活动内容”
  • 基于OpenResty的Lua Web框架lor0.0.2预览版发布
  • 理解IaaS, PaaS, SaaS等云模型 (Cloud Models)
  • 一道面试题引发的“血案”
  • 原生JS动态加载JS、CSS文件及代码脚本
  • # 计算机视觉入门
  • #Lua:Lua调用C++生成的DLL库
  • #Ubuntu(修改root信息)
  • (1)(1.13) SiK无线电高级配置(五)
  • (Oracle)SQL优化技巧(一):分页查询
  • (八十八)VFL语言初步 - 实现布局
  • (附源码)springboot优课在线教学系统 毕业设计 081251
  • (十二)springboot实战——SSE服务推送事件案例实现
  • (十五)devops持续集成开发——jenkins流水线构建策略配置及触发器的使用
  • (详细版)Vary: Scaling up the Vision Vocabulary for Large Vision-Language Models
  • (转)创业家杂志:UCWEB天使第一步
  • .NET 自定义中间件 判断是否存在 AllowAnonymousAttribute 特性 来判断是否需要身份验证
  • @拔赤:Web前端开发十日谈
  • [2008][note]腔内级联拉曼发射的,二极管泵浦多频调Q laser——
  • [2016.7 day.5] T2
  • [20171102]视图v$session中process字段含义
  • [20190401]关于semtimedop函数调用.txt
  • [android] 手机卫士黑名单功能(ListView优化)
  • [asp.net core]project.json(2)