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

Android图片缓存之Glide进阶

前言:

     前面学习了Glide的简单使用(Android图片缓存之初识Glide),今天来学习一下Glide稍微复杂一点的使用。

 图片缓存相关博客地址:

  • Android图片缓存之Bitmap详解
  • Android图片缓存之初识Glide
  • Android图片缓存之Glide进阶
  • Android图片缓存之Lru算法

GlideModule使用:

       GlideModule 是一个抽象方法,全局改变 Glide 行为的一个方式,通过全局GlideModule 配置Glide,用GlideBuilder设置选项,用Glide注册ModelLoader等。

  1.)自定义一个GlideModule 

public class MyGlideModule implements GlideModule {
    @Override public void applyOptions(Context context, GlideBuilder builder) {
        // Apply options to the builder here.
    }

    @Override public void registerComponents(Context context, Glide glide) {
        // register ModelLoaders here.
    }
}

  2.)AndroidManifest.xml注册

<manifest ...>
    <!-- ... permissions -->
    <application ...>
        <meta-data
            android:name="com.mypackage.MyGlideModule"
            android:value="GlideModule" />
        <!-- ... activities and other components -->
    </application>
</manifest>

  3.)添加混淆处理

-keepnames class com.mypackage.MyGlideModule
# or more generally:
#-keep public class * implements com.bumptech.glide.module.GlideModule

  4.)多个GlideModule冲突问题

   GlideModule不能指定调用顺序,所以应该避免不同的GlideModule之间有冲突的选项设置,可以考虑将所有的设置都放到一个GlideModule里面,或者排除掉某个manifest文件的某个Module,代码如下:

<meta-data android:name=”com.mypackage.MyGlideModule” tools:node=”remove” />

GlideBuilder设置选项:

1.)设置Glide内存缓存大小

 int maxMemory = (int) Runtime.getRuntime().maxMemory();//获取系统分配给应用的总内存大小
 int memoryCacheSize = maxMemory / 8;//设置图片内存缓存占用八分之一
 //设置内存缓存大小
 builder.setMemoryCache(new LruResourceCache(memoryCacheSize));

  获取默认的内存使用计算函数

MemorySizeCalculator calculator = new MemorySizeCalculator(context);  
int defaultMemoryCacheSize = calculator.getMemoryCacheSize();  
int defaultBitmapPoolSize = calculator.getBitmapPoolSize();  

2.)设置Glide磁盘缓存大小

 File cacheDir = context.getExternalCacheDir();//指定的是数据的缓存地址
 int diskCacheSize = 1024 * 1024 * 30;//最多可以缓存多少字节的数据
 //设置磁盘缓存大小
 builder.setDiskCache(new DiskLruCacheFactory(cacheDir.getPath(), "glide", diskCacheSize));

 也可以通过如下两种方式

  //存放在data/data/xxxx/cache/
  builder.setDiskCache(new InternalCacheDiskCacheFactory(context, "glide", diskCacheSize));
  //存放在外置文件浏览器
  builder.setDiskCache(new ExternalCacheDiskCacheFactory(context, "glide", diskCacheSize));

3.)设置图片解码格式

//设置图片解码格式
builder.setDecodeFormat(DecodeFormat.PREFER_ARGB_8888);

默认格式RGB_565使用内存是ARGB_8888的一半,但是图片质量就没那么高了,而且不支持透明度

4.)设置缓存内存大小

 //设置BitmapPool缓存内存大小
  builder.setBitmapPool(new LruBitmapPool(memoryCacheSize));

5.)设置一个用来检索cache中没有的Resource的ExecutorService

为了使缩略图请求正确工作,实现类必须把请求根据Priority优先级排好序。

builder.setDiskCacheService(ExecutorService service);
builder.setResizeService(ExecutorService service);

使用ModelLoader自定义数据源:

例如我们使用了七牛云存储,要根据不同的要求请求不同尺寸不同质量的图片,这时我们就可以使用自定义数据源

  1.)定义处理URL接口

public interface IDataModel {

    String buildDataModelUrl(int width, int height);

}

2.)实现处理URL接口

JpgDataModel
public class JpgDataModel implements IDataModel {
    private String dataModelUrl;

    public JpgDataModel(String dataModelUrl) {
        this.dataModelUrl = dataModelUrl;
    }

    @Override
    public String buildDataModelUrl(int width, int height) {
        //http://78re52.com1.z0.glb.clouddn.com/resource/gogopher.jpg?imageView2/1/w/200/h/200/format/jpg
        return String.format("%s?imageView2/1/w/%d/h/%d/format/jpg", dataModelUrl, width, height);
    }
}
WebpDataModel
public class WebpDataModel implements IDataModel {
    private String dataModelUrl;

    public WebpDataModel(String dataModelUrl) {
        this.dataModelUrl = dataModelUrl;
    }

    @Override
    public String buildDataModelUrl(int width, int height) {
        //http://78re52.com1.z0.glb.clouddn.com/resource/gogopher.jpg?imageView2/1/w/200/h/200/format/webp
        return String.format("%s?imageView2/1/w/%d/h/%d/format/webp", dataModelUrl, width, height);
    }
}

3.)实现ModelLoader

public class MyDataLoader extends BaseGlideUrlLoader<IDataModel> {
    public MyDataLoader(Context context) {
        super(context);
    }

    public MyDataLoader(ModelLoader<GlideUrl, InputStream> urlLoader) {
        super(urlLoader, null);
    }

    @Override
    protected String getUrl(IDataModel model, int width, int height) {
        return model.buildDataModelUrl(width, height);
    }

    /**
     */
    public static class Factory implements ModelLoaderFactory<IDataModel, InputStream> {

        @Override
        public ModelLoader<IDataModel, InputStream> build(Context context, GenericLoaderFactory factories) {
            return new MyDataLoader(factories.buildModelLoader(GlideUrl.class, InputStream.class));
        }

        @Override
        public void teardown() {
        }
    }
}

4.)根据不同的要求采用不同的策略加载图片

   //加载jpg图片
   Glide.with(this).using(new MyDataLoader(this)).load(new JpgDataModel(imageUrl)).into(imageView);
   //加载webp图片
   Glide.with(this).using(new MyDataLoader(this)).load(new WebpDataModel(imageUrl)).into(imageView);

5.)如何跳过.using()

public class MyGlideModule implements GlideModule {
    ...
    @Override
    public void registerComponents(Context context, Glide glide) {
        glide.register(IDataModel.class, InputStream.class, 
            new MyUrlLoader.Factory());
    }
}

上面的实现跳过using()

   //加载jpg图片
   Glide.with(this).load(new JpgDataModel(imageUrl)).into(imageView);
   //加载webp图片
   Glide.with(this).load(new WebpDataModel(imageUrl)).into(imageView);

 使用signature()实现自定义cacheKey:

    Glide 以 url、viewwidth、viewheight、屏幕的分辨率等做为联合key,官方api中没有提供删除图片缓存的函数,官方提供了signature()方法,将一个附加的数据加入到缓存key当中,多媒体存储数据,可用MediaStoreSignature类作为标识符,会将文件的修改时间、mimeType等信息作为cacheKey的一部分,通过改变key来实现图片失效相当于软删除。

1.)使用StringSignature

Glide.with(this).load(yourFileDataModel).signature(new StringSignature("1.0.0")).into(imageView);

2.)使用MediaStoreSignature

Glide.with(this) .load(mediaStoreUri).signature(new MediaStoreSignature(mimeType, dateModified, orientation)).into(view);

3.)使用自定义Signature

public class IntegerVersionSignature implements Key {
  private int currentVersion;
  public IntegerVersionSignature(int currentVersion) {
     this.currentVersion = currentVersion;
  }
  @Override
  public boolean equals(Object o) {
    if (o instanceof IntegerVersionSignature) {
      IntegerVersionSignature other = (IntegerVersionSignature) o;
      return currentVersion = other.currentVersion;
    }
    return false;
  }
  @Override
  public int hashCode() {
    return currentVersion;
  }
  @Override
  public void updateDiskCacheKey(MessageDigest md) {
    messageDigest.update(ByteBuffer.allocate(Integer.SIZE)
.putInt(signature).array());
  }
}

小结:

     今天学了Glide的一些自定义扩展知识,接下来学习一下Glide与OKHttp的结合,已经简单探析一下内部使用。

 

相关文章:

  • css(二):高度与文档流,line-box与box,宽度,position与
  • SQL Server2005作业执行失败的解决办法
  • 进度条14
  • 阿里云RDS PostgreSQL GPU加速规格(支持GIS时空加速)发布
  • 怎么隐藏MathType标尺
  • 用Python监听邻居家小姐姐的上网行为
  • Git同步原始仓库到Fork仓库中
  • 如何将PDF图片旋转90度,盘点这个小方法
  • php 0613框架基础
  • debian手册摘要
  • jQuery EasyUI使用教程之添加节点到树形菜单
  • 万物有源_Appium框架原理浅析
  • PHPDoc/PHPDocumentor生成API文档
  • 面试必问Elasticsearch倒排索引原理
  • Linux screen 解决会话终止当前进程断开的问题
  • [iOS]Core Data浅析一 -- 启用Core Data
  • 【React系列】如何构建React应用程序
  • CEF与代理
  • CSS3 聊天气泡框以及 inherit、currentColor 关键字
  • Django 博客开发教程 16 - 统计文章阅读量
  • ESLint简单操作
  • Java 多线程编程之:notify 和 wait 用法
  • Java知识点总结(JDBC-连接步骤及CRUD)
  • markdown编辑器简评
  • Mithril.js 入门介绍
  • MySQL常见的两种存储引擎:MyISAM与InnoDB的爱恨情仇
  • php面试题 汇集2
  • vue总结
  • XML已死 ?
  • 简单数学运算程序(不定期更新)
  • 排序(1):冒泡排序
  • 什么软件可以提取视频中的音频制作成手机铃声
  • 我这样减少了26.5M Java内存!
  • hi-nginx-1.3.4编译安装
  • 如何用纯 CSS 创作一个菱形 loader 动画
  • #pragma 指令
  • #经典论文 异质山坡的物理模型 2 有效导水率
  • (26)4.7 字符函数和字符串函数
  • (C#)Windows Shell 外壳编程系列9 - QueryInfo 扩展提示
  • (Redis使用系列) SpringBoot 中对应2.0.x版本的Redis配置 一
  • (附源码)ssm高校实验室 毕业设计 800008
  • (欧拉)openEuler系统添加网卡文件配置流程、(欧拉)openEuler系统手动配置ipv6地址流程、(欧拉)openEuler系统网络管理说明
  • (区间dp) (经典例题) 石子合并
  • (原创)攻击方式学习之(4) - 拒绝服务(DOS/DDOS/DRDOS)
  • (转)h264中avc和flv数据的解析
  • (轉貼) 資訊相關科系畢業的學生,未來會是什麼樣子?(Misc)
  • . NET自动找可写目录
  • .locked1、locked勒索病毒解密方法|勒索病毒解决|勒索病毒恢复|数据库修复
  • .Net MVC + EF搭建学生管理系统
  • .net wcf memory gates checking failed
  • .NET 线程 Thread 进程 Process、线程池 pool、Invoke、begininvoke、异步回调
  • .NetCore 如何动态路由
  • .sdf和.msp文件读取
  • .xml 下拉列表_RecyclerView嵌套recyclerview实现二级下拉列表,包含自定义IOS对话框...
  • @property @synthesize @dynamic 及相关属性作用探究