原文地址:https://github.com/bumptech/glide/wiki/Caching-and-Cache-Invalidation
一、缓存Keys:
Glide中的缓存keys由四部分构成:
1.DataFetcher的getId()方法返回的字串。一般DataFetchers只是返回数据模型的toString()方法的返回值,例如:如果你提供的是URL则返回URL的路径,如果提供的是文件则返回文件路径
2.override(int, int)方法中传入的高、宽值(如果调用了此方法),默认返回Target的getSize()方法提供的高、宽值
3.各个用于加载、缓存图片的编、解码器中的getId()方法返回的字串。只有当编、解码器影响到检索数据时才能包含id。例如:一个编码器只是简单地将byte数组写入磁盘,则此编码器是没有id值的,因为它没有对数据造成任何影响
4.每次加载时用到的一个可选的签名。
这些keys将以特定的顺序hash化,以用来创建唯一且安全的文件名用来将特定的图片缓存到磁盘。
二、缓存无效化:
由于缓存图片的文件名都是hash后的值,因此没有一个好的方式用来删除给定url或文件路径对应的磁盘上的全部缓存文件。因为Glide会同时缓存预览图、各种转码后的图片,每一种图片都会在缓存中创建一个新的文件,因此追踪并删除一个图片的所有缓存文件是很困难的。因此清楚缓存最好的方式是在内容改变时(url、uri、file path等)改变标识符。
三、自定义缓存无效化:
基于改变标识符的困难性,Glide提供了signature()方法用来将可控制的附加数据(签名)加入缓存key中。签名在存储多媒体资源时效果很好,同样,缓存其他内容时也可以在其中保留一些元数据。
1.缓存多媒体资源:可以使用Glide的MediaStoreSignature作为签名。MediaStoreSignature允许你将修改时间、mimeType、多媒体的定位这三种信息混入缓存key。
2.文件:可以使用StringSignature来混入文件的修改时。。
3.Urls:最好地清除urls缓存的方式是服务端改变url并在url改变时在客户端进行更新。不过,同样可以使用StringSignature混入任意元数据(例如版本号)的方式来取代。
传入string signatures:
Glide.with(yourFragment)
.load(yourFileDataModel)
.signature(new StringSignature(yourVersionMetadata))
.into(yourImageView);
多媒体签名:
Glide.with(fragment)
.load(mediaStoreUri)
.signature(new MediaStoreSignature(mimeType, dateModified, orientation))
.into(view);
通过实现Key接口可以自定义签名(确保实现equals()、hashCode()以及updateDiskCacheKey()方法):
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 messageDigest) {
messageDigest.update(ByteBuffer.allocate(Integer.SIZE).putInt(currentVersion).array());
}
}
public class CustomFileSignature implements Key {
private final String fileName;
private final long dateModified;
private final String filePath;
public CustomFileSignature(String fileName, long dateModified, String filePath) {
this.fileName = fileName;
this.dateModified = dateModified;
this.filePath = filePath;
}
@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (o == null || getClass() != o.getClass()) {
return false;
}
CustomFileSignature that = (CustomFileSignature) o;
if (fileName != null ? !fileName.equals(that.fileName) : that.fileName != null) {
return false;
}
if (dateModified != that.dateModified) {
return false;
}
if (filePath != null ? !filePath.equals(that.filePath) : that.filePath != null) {
return false;
}
return true;
}
@Override
public int hashCode() {
int result = fileName != null ? fileName.hashCode() : 0;
// dateModified >>> 32 ——> dateModified / ((int) Math.pow(2, 32))
result = 31 * result + (int) (dateModified ^ (dateModified >>> 32));
result = 31 * result + (filePath != null ? filePath.hashCode() : 0);
return result;
}
@Override
public void updateDiskCacheKey(MessageDigest messageDigest) throws UnsupportedEncodingException {
byte[] data = ByteBuffer.allocate(12)
.putLong(dateModified)
.array();
messageDigest.update(data);
messageDigest.update(fileName.getBytes(STRING_CHARSET_NAME));
messageDigest.update(filePath.getBytes(STRING_CHARSET_NAME));
}
}
注意:为了不影响性能,最好在后台加载元数据
也可以使用diskCacheStrategy()以及DiskCacheStrategy.NONE的方式来彻底禁用磁盘缓存。
Glide.with(context).load(imageUrl).diskCacheStrategy(DiskCacheStrategy.NONE).into(view);