★40.Glide

前言

在写本教程时, Glide 的版本号是3.7。

简介

  • 相关网址:GitHub地址英文文档
  • Glide 是一个媒体管理框架,支持图片、原生视频的加载,包括从网络上加载。

相关概念

  • Target:可以指ViewTarget(含View)或SimpleTarget(不含View)。
  • Signature:签名,可以唯一地标识一个对象。
  • Resource.recycle():表示该资源不被引用,可以被回收。

简单示例

创建Glide

Glide.with(this)
        .load(/* 加载源 */)
        .into(imageView);

Glide的简单配置项

  • 设置绑定的生命周期:with(this)
  • 设置加载资源:load(/* 加载源 */),加载源可以是如下几种类型对象:
    • byte[]:字节数组。
    • String:可以是文件路径或者Url等。
    • Integer:资源id。
    • File:文件。
    • Uri:数据。
  • 设置加载的优先级:priority(/* 优先级 */)
    • Priority.LOW
    • Priority.NORMAL
    • Priority.HIGH
    • Priority.IMMEDIATE
  • 设置“加载中”占位图片:placeholder(/* 加载源 */)
  • 设置“加载失败”占位图片:error(/* 加载源 */)
  • 设置gif为静态:asBitmap()
  • 设置gif为动态:asGif()
  • 设置加载动画为无动画:dontAnimate()
  • 设置加载动画为淡入淡出动画:crossFade(/* 动画时间 */)
  • 设置加载动画为自定义动画:animate(R.anim.id)
  • 设置尺寸:override(800, 800)
  • 设置裁剪策略为拉伸裁剪:centerCrop()
  • 设置裁剪策略为拉伸不裁剪:fitCenter()

Glide的复杂配置项

缩略图

简介

  • 缩略图 :在图片加载过程中显示的图片。

简单的缩略图加载

  • 简单的 缩略图 是和 原图 一起加载的。
.thumbnail(0.1f)

独立的缩略图加载

  • 缩略图 的加载是和 原图 的加载相互独立的。
  • 以下代码可以多次递归,即设置 原图缩略图缩略图 又设置其 缩略图 ,可实现从模糊到清晰的加载过程。
DrawableRequestBuilder<String> thumbnailRequest = Glide
        .with(context)
        .load(eatFoodyImages[2]);
.thumbnail(thumbnailRequest)

错误调试

简介

  • RequestListener.onException(...):资源加载出错时调用。
  • RequestListener.onResourceReady(...):资源加载完成时调用。

1. 创建RequestListener

RequestListener<String, GlideDrawable> errorListener = new RequestListener<String, GlideDrawable>() {
    @Override
    public boolean onException(Exception e, String model, Target<GlideDrawable> target,
                               boolean isFirstResource) {
        Log.e("onException", e.toString() + "  model:" + model + " isFirstResource: " + isFirstResource);
        imageView.setImageResource(R.mipmap.ic_launcher);
        return false;
    }

    @Override
    public boolean onResourceReady(GlideDrawable resource, String model, Target<GlideDrawable> target,
                                   boolean isFromMemoryCache, boolean isFirstResource) {
        Log.e("onResourceReady", "isFromMemoryCache:" + isFromMemoryCache + "  model:" + model
                + " isFirstResource: " + isFirstResource);
        return false;
    }
};

2. 绑定RequestListener

.listener(errorListener)

加载图片到通知

  • 先创建一个NotificationTarget对象,再load(NotificationTarget)即可实现目标。具体使用方法看NotificationTarget构造方法。

加载图片到应用小控件

  • 先创建一个AppWidgetTarget对象,再load(AppWidgetTarget)即可实现目标。具体使用方法看AppWidgetTarget构造方法。

缓存

简介

  • Glide 默认使用 内存缓存磁盘缓存 来避免不必要的网络请求。

缓存策略

  • 跳过 内存缓存skipMemoryCache(true)
  • 跳过 磁盘缓存diskCacheStrategy(/* 磁盘缓存策略 */)
    • DiskCacheStrategy.ALL:如果你请求一个 1000x1000像素 的图片,你的ImageView500x500像素Glide 会缓存两种尺寸的图片。
    • DiskCacheStrategy.NONE
    • DiskCacheStrategy.SOURCE:只缓存图片转换前的数据。
    • DiskCacheStrategy.RESULT:只缓存图片转换后的数据。如果你请求一个 1000x1000像素 的图片,你的ImageView500x500像素 ,则只缓存后者。

缓存失效【Todo】

官方教程

自定义Target

SimpleTarget

简介

  • SimpleTarget泛型 参数可以是BitmapDrawable
  • SimpleTarget不可用于自定义 View
  • 可以通过SimpleTarget获取BitmapDrawable对象,手动加载到需要的地方。
  • 不要把SimpleTarget对象定义为 匿名内部类 ,而是定义为 类属性 ,以避免被Android回收机制回收,从而永远不会进行回调。

1. 创建SimpleTarget

SimpleTarget target = new SimpleTarget<Bitmap>(/* 长 */, /* 宽 */) {
    @Override
    public void onResourceReady(Bitmap bitmap, GlideAnimation glideAnimation) {
        // 获取到bitmap
        imageView1.setImageBitmap(bitmap);
    }
};

2. 绑定SimpleTarget

// noinspection unchecked
Glide
        .with(context)      // 如果是在Activity生命周期外请求的话,需要使用application context。
        .load(url)
        .asBitmap()         // 通过asBitmap(),避免当资源是视频等,会出现未知格式的情况。
        .into(target);

ViewTarget

简介

  • ViewTarget可用于自定义 View
  • ViewTargetSimpleTarget的区别是ViewTarget内部含有 View

1. 创建ViewTarget

ViewTarget<MyView, GlideDrawable> viewTarget = new ViewTarget<MyView, GlideDrawable>(customView) {
    @Override
    public void onResourceReady(GlideDrawable resource, GlideAnimation<? super GlideDrawable> glideAnimation) {
        this.view.setImage(resource.getCurrent());
    }
};

2. 绑定ViewTarget

.into(target)

自定义BitmapTransformation

1. 定义BitmapTransformation

public class BlurTransformation extends BitmapTransformation {
    private RenderScript rs;

    public BlurTransformation(Context context) {
        super(context);
        rs = RenderScript.create(context);
    }

    @Override
    protected Bitmap transform(BitmapPool pool, Bitmap toTransform, int outWidth, int outHeight) {
        Bitmap blurredBitmap = toTransform.copy(Bitmap.Config.ARGB_8888, true);

        // 分配内存
        Allocation input = Allocation.createFromBitmap(
                rs,
                blurredBitmap,
                Allocation.MipmapControl.MIPMAP_FULL,
                Allocation.USAGE_SHARED
        );
        Allocation output = Allocation.createTyped(rs, input.getType());

        // Load up an instance of the specific script that we want to use.
        ScriptIntrinsicBlur script = ScriptIntrinsicBlur.create(rs, Element.U8_4(rs));
        script.setInput(input);

        // Set the blur radius
        script.setRadius(10);

        // Start the ScriptIntrinisicBlur
        script.forEach(output);

        // Copy the output to the blurred bitmap
        output.copyTo(blurredBitmap);

        toTransform.recycle();
        return blurredBitmap;
    }

    @Override
    public String getId() {
        return "blur";
    }
}

2. 应用变换

  • 方式一:
    .transform(new BlurTransformation(context))
    
  • 方式二:
    .bitmapTransform(new BlurTransformation(context))
    
  • 方式三(多重变换):
    .transform(new GreyscaleTransformation(context), new BlurTransformation(context))
    

Glide Modules

简介

  • 通过实现GlideModule接口并在 AndroidManifest.xml 中注册来实现。
  • 可以同时注册多个GlideModule类,不过要确保没有冲突。
  • 如果想要禁止GlideModule,只要从 AndroidManifest.xml 里移除它。

1. 定义GlideModules

GlideModules外壳

public class SimpleGlideModule implements GlideModule {
    @Override public void applyOptions(Context context, GlideBuilder builder) {
        // todo
    }

    @Override public void registerComponents(Context context, Glide glide) {
        // todo
    }
}

重写applyOptions()

1. 自定义内存缓存

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

int customMemoryCacheSize = (int) (1.2 * defaultMemoryCacheSize);
int customBitmapPoolSize = (int) (1.2 * defaultBitmapPoolSize);

builder.setMemoryCache(new LruResourceCache(customMemoryCacheSize));
builder.setBitmapPool(new LruBitmapPool(customBitmapPoolSize));

2. 自定义磁盘缓存

int cacheSize100MegaBytes = 104857600;
builder.setDiskCache(new InternalCacheDiskCacheFactory(context, cacheSize100MegaBytes));
// builder.setDiskCache(new ExternalCacheDiskCacheFactory(context, cacheSize100MegaBytes));
int cacheSize100MegaBytes = 104857600;
String downloadDirectoryPath = Environment.getDownloadCacheDirectory().getPath();
builder.setDiskCache(new DiskLruCacheFactory(downloadDirectoryPath, cacheSize100MegaBytes));
// 子目录
// builder.setDiskCache(new DiskLruCacheFactory(downloadDirectoryPath, "glidecache", cacheSize100MegaBytes));

3. 自定义图片质量

builder.setDecodeFormat(DecodeFormat.PREFER_ARGB_8888);
// builder.setDecodeFormat(DecodeFormat.PREFER_RGB_565);

4. 其他

  • GlideBuilder.setDiskCacheService():设置请求的资源在 磁盘缓存 里的时候需要执行的ExecutorService
  • GlideBuilder.setResizeService():设置请求的资源不在 磁盘缓存 里的时候需要执行的ExecutorService

重写registerComponents()

使用OkHttp实现从Https中加图片

工具类:OkHttpUrlLoaderOkHttpStreamFetcher

glide.register(GlideUrl.class, InputStream.class, new OkHttpUrlLoader.Factory(httpsOkHttpClient));

2. 注册

AndroidManifest.xml 中:

<application>
    ...
    <meta-data
            android:name=".SimpleGlideModule"
            android:value="GlideModule"/>
</application>

Glide Transformations

简介

简单示例

单个转换

Glide.with(this).load(R.drawable.demo)
        .bitmapTransform(new BlurTransformation(context))
        .into((ImageView) findViewById(R.id.image));

多个转换

Glide.with(this).load(R.drawable.demo)
        .bitmapTransform(new BlurTransformation(context, 25), new CropCircleTransformation(context))
        .into((ImageView) findViewById(R.id.image));

Crop

  • CropTransformation
  • CropCircleTransformation
  • CropSquareTransformation
  • RoundedCornersTransformation

Color

  • ColorFilterTransformation
  • GrayscaleTransformation

Blur

  • BlurTransformation

Mask(掩模)

  • MaskTransformation

GPU Filter(滤镜)

  • ToonFilterTransformation
  • SepiaFilterTransformation
  • ContrastFilterTransformation
  • InvertFilterTransformation
  • PixelationFilterTransformation
  • SketchFilterTransformation
  • SwirlFilterTransformation
  • BrightnessFilterTransformation
  • KuwaharaFilterTransformation
  • VignetteFilterTransformation
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,245评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,749评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,960评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,575评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,668评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,670评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,664评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,422评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,864评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,178评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,340评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,015评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,646评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,265评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,494评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,261评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,206评论 2 352

推荐阅读更多精彩内容