• 学习Glide 常用场景的写法 +


    transform 用来完成剪裁(切圆)或对位图应用过滤器,但它也可以用于转换GIF动画,甚至自定义的资源类型。

    使用场景的写法

    默认加载图

    				Glide.with(this).load(uri).into(imageView);
    
    • 1

    加载本地图

                    Glide.with(this).clear(imageView);
                    Glide.with(this).load(R.drawable.bg).into(imageView);
    
    • 1
    • 2

    圆图

                    Glide.with(this).clear(imageView);
                    Glide.with(this).load(uri).circleCrop().into(imageView);
    
    • 1
    • 2

    站位图

                    RequestOptions requestOptions = new RequestOptions()
                            .placeholder(R.mipmap.ic_del)
                            .error(R.mipmap.leak_canary_icon)
                            .fallback(R.mipmap.tv_title_all_music);
    
                    Glide.with(this).load(uri).apply(requestOptions).into(imageView);
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    1. placeholder
      正在请求图片的时候展示的图片
    2. error
      如果请求失败的时候展示的图片 (如果没有设置,还是展示placeholder的占位符)
    3. fallback
      如果请求的url/model为 null 的时候展示的图片 (如果没有设置,还是展示placeholder的占位符)

    改变图片大小

    改变图片的大小不是改变View的大小~o

                    GlideApp.with(getActivity())
                            .load(uri)
                            .centerCrop()
                            .override(imageWidthSize,imageHeightSize)
                            .into(imageView);
    
    • 1
    • 2
    • 3
    • 4
    • 5

    如果View所需的图片大小和override设置的不一致,override设置后的图片将会被压缩或者拉伸。

    获取Bitmap

                    FutureTarget<Bitmap> target = Glide.with(this)
                            .asBitmap()
                            .load(uri)
                            .submit();
                    new Thread(new Runnable() {
                        @Override
                        public void run() {
                            try {
                                Bitmap bitmap = target.get();//同步请求
                                runOnUiThread(new Runnable() {
                                    @Override
                                    public void run() {
                                        Glide.with(getActivity()).clear(imageView);
                                        Glide.with(getActivity()).load(bitmap).into(imageView);
                                    }
                                });
                            } catch (Exception e) {
                                e.printStackTrace();
                            }
                        }
                    }).start();
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21

    图形变换

    glide-transformations https://github.com/wasabeef/glide-transformations
    :一个 Android 转换库,为Glide提供各种图像转换。下面是效果图:

    在这里插入图片描述

    Crop
    CropTransformation 自定义矩形剪裁
    CropCircleTransformation 圆形
    CropCircleWithBorderTransformation 圆形描边
    CropSquareTransformation 正方形剪裁
    RoundedCornersTransformation 圆角剪裁

    Color
    ColorFilterTransformation 颜色滤镜
    GrayscaleTransformation 灰度级转换

    Blur
    BlurTransformation 模糊处理

    Mask
    MaskTransformation 遮罩掩饰(视图叠加处理)

    GPU Filter (use GPUImage)
    Will require add dependencies for GPUImage. GPU过滤(需要依赖GPUImage库)

    原图:

    ToonFilterTransformation 卡通滤波器
    SepiaFilterTransformation 乌墨色滤波器
    ContrastFilterTransformation 对比度滤波器
    InvertFilterTransformation 反转滤波器

    PixelationFilterTransformation 像素化滤波器
    SketchFilterTransformation 素描滤波器
    SwirlFilterTransformation 旋转滤波器
    BrightnessFilterTransformation 亮度滤波器

    KuwaharaFilterTransformation Kuwahara滤波器
    VignetteFilterTransformation 装饰图滤波器

    添加依赖

      implementation 'jp.wasabeef:glide-transformations:4.3.0'
      // If you want to use the GPU Filters
      implementation 'jp.co.cyberagent.android:gpuimage:2.1.0'
    
    • 1
    • 2
    • 3


    Android之Glide图像处理:https://blog.csdn.net/qq_36347817/article/details/84983198

    Glide的变换(美图效果): https://blog.csdn.net/weixin_38364803/article/details/78603993

                    Glide.with(this)
                            .load(uri)
    //                        .transform(new CropTransformation(150,50, CropTransformation.CropType.TOP))
    //                        .transform(new CropCircleTransformation())
    //                        .transform(new CropSquareTransformation())
    //                        .transform(new ColorFilterTransformation(0x7900CCCC))
    //                        .transform(new BlurTransformation(20, 1))
    //                        .transform(new CropCircleWithBorderTransformation())
    //                        .transform(new ToonFilterTransformation( 0.2F, 10F))
    //                        .transform(new SepiaFilterTransformation(1.0F))
    //                        .transform(new ContrastFilterTransformation(3F))
    //                        .transform(new InvertFilterTransformation())
                            .transform(new PixelationFilterTransformation(20F))
                            .transform(new SketchFilterTransformation())
                            .transform(new SwirlFilterTransformation( 1.0F, 0.4F, new PointF(0.5F, 0.5F)))
                            .transform(new BrightnessFilterTransformation( 0.5F))
                            .transform(new KuwaharaFilterTransformation( 10))
                            .transform(new VignetteFilterTransformation( new PointF(0.5F, 0.5F), new float[]{0.0F, 0.0F, 0.0F}, 0.0F, 0.5F))
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18

    磁盘缓存 (Disk Cache)

    Glide 使用 DiskLruCacheWrapper 作为默认的磁盘缓存。
    DiskLruCacheWrapper 是一个使用 LRU 算法的固定大小的磁盘缓存。默认磁盘大小为 250 MB ,位置是在应用的 缓存文件夹 中的一个 特定目录 。

    public interface DiskCache {
    
      /** An interface for lazily creating a disk cache. */
      interface Factory {
        /** 250 MB of cache. */
        int DEFAULT_DISK_CACHE_SIZE = 250 * 1024 * 1024;
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    缓存类型

    资源类型(Resource) - 该图片是否之前曾被解码、转换并写入过磁盘缓存?
    数据来源 (Data) - 构建这个图片的资源是否之前曾被写入过文件缓存?

    缓存策略

    DiskCacheStrategy.NONE:表示不缓存任何内容;
    DiskCacheStrategy.RESOURCE:在资源解码后将数据写入磁盘缓存,即经过缩放等转换后的图片资源;
    DiskCacheStrategy.DATA:在资源解码前将原始数据写入磁盘缓存;
    DiskCacheStrategy.ALL :使用DATA和RESOURCE缓存数据;
    DiskCacheStrategy.AUTOMATIC:
    它会尝试对本地和远程图片使用最佳的策略。
    当加载远程数据时,AUTOMATIC 策略仅会存储未被加载过程修改过的原始数据,因为下载远程数据相比调整磁盘上已经存在的数据要昂贵得多;
    对于本地数据,AUTOMATIC 策略则会仅存储变换过的缩略图,因为即使需要再次生成另一个尺寸或类型的图片,取回原始数据也很容易。默认使用这种缓存策略。

    • 关闭磁盘缓存
                    Glide.with(this)
                            .load(uri)
                            .diskCacheStrategy(DiskCacheStrategy.NONE)
                            .into(imageView);
    
    • 1
    • 2
    • 3
    • 4
    • 配置默认缓存路径为外部存储路径
    @GlideModule
    public class GlideConfigModule extends AppGlideModule {
      @Override
      public void applyOptions(Context context, GlideBuilder builder) {
        builder.setDiskCache(new ExternalCacheDiskCacheFactory(context));
      }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 配置默认缓存路径为内部存储路径
        builder.setDiskCache(new InternalCacheDiskCacheFactory(context));
    
    • 1
    • 配置默认缓存大小
        builder.setDiskCache(new InternalCacheDiskCacheFactory(context,100*1024*1024));
    
    • 1
    • 配置默认缓存大小和文件夹名
        builder.setDiskCache(new InternalCacheDiskCacheFactory(context,name,100*1024*1024));
    
    • 1
    • 自行实现 DiskCache.Factory
    @GlideModule
    public class MyGlideModule extends AppGlideModule {
      @Override
      public void applyOptions(Context context, GlideBuilder builder) {
        builder.setDiskCache(new DiskCache.Factory() {
            @Override
            public DiskCache build() {
              return new MyCustomDiskCache();
            }
        });
      }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    或者

        @Override
        public void applyOptions(@NonNull final Context context, GlideBuilder builder) {
            builder.setDiskCache(() -> {
                File cacheLocation = new File(context.getExternalCacheDir(), "GlideCache");
                if (!cacheLocation.exists()) {
                    boolean result = cacheLocation.mkdirs();
                    AppLogUtils.i(TAG, "create glide image cache dir result == " + result);
                } else {
                    AppLogUtils.i(TAG, "glide image cache dir is already exists");
                }
                return DiskLruCacheWrapper.create(cacheLocation, 100*1024*1024);
            });
        }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 清理磁盘缓存
    new AsyncTask<Void, Void, Void> {
      @Override
      protected Void doInBackground(Void... params) {
        // This method must be called on a background thread.
        Glide.get(applicationContext).clearDiskCache();
        return null;
      }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 仅从缓存加载图片

    某些情形下,你可能希望只要图片不在缓存中则加载直接失败(比如省流量模式?–译者注)。如果要完成这个目标,你可以在单个请求的基础上使用 onlyRetrieveFromCache 方法:

    Glide.with(fragment)
      .load(url)
      .onlyRetrieveFromCache(true)
      .into(imageView);
    
    • 1
    • 2
    • 3
    • 4

    如果图片在内存缓存或在磁盘缓存中,它会被展示出来。否则只要这个选项被设置为 true ,这次加载会视同失败。

    内存缓存 (Memory cache)

    Glide使用 LruResourceCache ,这是 MemoryCache 接口的一个缺省实现,使用固定大小的内存和 LRU 算法。LruResourceCache 的大小由 Glide 的 MemorySizeCalculator 类来决定,这个类主要关注设备的内存类型,设备 RAM 大小,以及屏幕分辨率。

    缓存类型

    活动资源 (Active Resources) - 现在是否有另一个 View 正在展示这张图片?
    内存缓存 (Memory cache 具体为 LruResourceCache) - 该图片是否最近被加载过并仍存在于内存中?

    从内存缓存(LruResourceCache)中拿到资源时候就主动添加到活动缓存(ActiveResource)中,并清理Lru内存缓存(LruResourceCache)中的资源,这样做的好处就是是保护正在使用资源不被Lru算法回收掉。

    ActiveResources是一个弱引用的HashMap,用来缓存正在使用的图片,保存这个图片不会被Lru算法回收掉。图片用完之后会重新添加到Lru内存缓存中。

    ActiveResources和LruResourceCache是内存缓存,属于运行时缓存且互斥(同一张图片不会同时缓存在ActiveResources和LruResourceCache中),应用被杀死后内存缓存将不存在。

    设置缓存

    • 计算大小
    @GlideModule
    public class MyGlideModule extends AppGlideModule {
        @Override
        public void applyOptions(Context context, GlideBuilder builder) {
            MemorySizeCalculator calculator = new MemorySizeCalculator.Builder(context)
                    .build();
                    
    		AppLogUtils.i(TAG, "getMemoryCacheSize =="+calculator.getMemoryCacheSize());
            builder.setMemoryCache(new LruResourceCache(calculator.getMemoryCacheSize()));
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    系统:HarmonyOS
    型号:HUAWEI P30
    系统版本:10
    运行内存:8GB

    在此手机上运行:
    getMemoryCacheSize = 19569600 约等 18MB

    • 设置大小
        @Override
        public void applyOptions(@NonNull final Context context, GlideBuilder builder) {
    
            int memoryCacheSizeBytes = new MemorySizeCalculator.Builder(context)
                    .build().getMemoryCacheSize();
    
    //        memoryCacheSizeBytes = new MemorySizeCalculator.Builder(context)
    //                 .setMemoryCacheScreens(2)
    //                 .build().getMemoryCacheSize();
    
    //         memoryCacheSizeBytes = 1024 * 1024 * 10;
    
            builder.setMemoryCache(new LruResourceCache(1.2 * memoryCacheSizeBytes));
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 跳过内存缓存
      .skipMemoryCache(true)  //true是跳过缓存
    
    • 1
    • 清理内存缓存
    	Glide.get(this).clearMemory();
    
    • 1

    BitmapPool又是何用?

    转:https://www.jianshu.com/p/bfc3f1f463c8

    复用创建过的Bitmap对象。

    • 写入:内存缓存溢出时、EngineResource.release()至引用数为0和ActiveResources被GC回收后如果不使用内存缓存,会直接将其放入bitmapPool等待复用。

    • 读取:需要创建Bitmap的地方,如果复用池已存在可用的bitmap则直接返回,没有则创建新的bitmap返回。

    • 剔除:超过size后溢出、不支持复用直接剔除。

    当一个Bitmap从内存缓存 被动 的被移除(内存紧张、达到maxSize)的时候并不会被recycle。

    而是加入这个BitmapPool,只有从这个BitmapPool 被动被移除的时候,Bitmap的内存才会真正被recycle释放。

    • 计算大小
        @Override
        public void applyOptions(@NonNull final Context context, GlideBuilder builder) {
    
            int bitmapCacheSizeBytes =  new MemorySizeCalculator.Builder(context)
                    .build().getBitmapPoolSize();
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    系统:HarmonyOS
    型号:HUAWEI P30
    系统版本:10
    运行内存:8GB

    在此手机上运行:
    getBitmapPoolSize= 19569600 约等 18MB 和内存缓存的大小一致。

    • 设置大小
            int bitmapCacheSizeBytes =  new MemorySizeCalculator.Builder(context)
                    .build().getBitmapPoolSize();
    
            AppLogUtils.i(TAG, "getBitmapPoolSize ==" + memoryCacheSizeBytes);
            
            builder.setBitmapPool(new LruBitmapPool((long) (1.2 * bitmapCacheSizeBytes)));
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    如何在Url可能变化的场景下使用缓存?

    Glide高级使用技巧解决缓存key问题,防止反复加载:https://www.it610.com/article/1303909044058099712.htm

    Glide使用高级技巧(解决Glide生成缓存Key问题):http://quanzhan.applemei.com/webStack/TWpnek9BPT0%3D

    场景是:Url地址部分变化(Url = Url + Token),Toekn一直变化,但是图片不会变,问如何使用缓存?

    重写 getCacheKey

    import com.bumptech.glide.load.model.GlideUrl;
    
    public class MyGlideUrl extends GlideUrl {
    
        private String eventId;
    
        public MyGlideUrl(String url,String eventId) {
            super(url);
            this.eventId = eventId;
        }
    
        @Override
        public String getCacheKey() {
            return eventId;
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    MyGlideUrl myGlideUrl = new MyGlideUrl(picUrl,event.getId());
    Glide.with(cext)
        .load(myGlideUrl)
        .apply(requestOptions)
        .into(holder.eventIv);
    
    • 1
    • 2
    • 3
    • 4
    • 5

    参考地址

    Android Glide 的简单使用(一):https://blog.csdn.net/g984160547/article/details/119991748

    Glide缓存机制:https://blog.csdn.net/xingyu19911016/article/details/125442418

    https://muyangmin.github.io/glide-docs-cn/doc/caching.html

    提问Glide - 缓存篇:https://www.jianshu.com/p/bfc3f1f463c8

  • 相关阅读:
    MySQL如何改进LRU算法
    C++ 线程安全注解
    【TypeScript】学习笔记(二)
    Python 基础30道测试题
    PAT 1048 Find Coins
    Linux---进程(1)
    docker安装mysql
    SpringCloud Alibaba - Seata 四种分布式事务解决方案(XA、AT)+ 实践部署(上)
    ubuntu 18 断网/网络不通,利用 ifconfig -a查看只有lo
    java基础---缓冲流、数据流、打印流
  • 原文地址:https://blog.csdn.net/weixin_35691921/article/details/120865058