地緩存
在緩存文件時對文件名稱的修改提供了兩種方式,每一種方式對應了一個Java類
1) HashCodeFileNameGenerator,該類負責獲取文件名稱的hashcode然後轉換成字符串。
2) Md5FileNameGenerator,該類把源文件的名稱同過md5加密後保存。
兩個類都繼承了FileNameGenerator接口
在DefaultConfigurationFactory類中提供了一個工廠方法createFileNameGenerator,該方法返回了一個默認的FileNameGenerator對象:HashCodeFileNameGenerator.
public static FileNameGenerator createFileNameGenerator() {
return new HashCodeFileNameGenerator();
}
實現
首先定義了DiscCacheAware接口,該接口提供了如下方法
File getFileDectory() 返回磁盤緩存的根目錄
File get(String imageUri) 根據uri從緩存中獲取圖片
boolean save(String imageUri,InputStream iamgeStream,IoUtils.CopyListener listener) 把圖片保存在磁盤緩存上
boolean save(String imageUri,Bitmap bitmap) 保存bitmap對象到磁盤緩存上
boolean remove(imageUri) 根據imageUri刪除文件
void close() 關閉磁盤緩存,釋放資源
void clear() 清空磁盤緩存
然後定義了另外一個沒方法的接口DiskCache,該接口只是簡單的繼承了DiscCacheAware接口。
BaseDiscCache實現了DiskCache,該類是個抽象類,該類定義了磁盤緩沖區的以下的屬性:
1) 默認的緩存大小為32k
2) 默認壓縮後的圖片格式為PNG(作為Bitmap的compress方法的第一個參數)
3) 默認壓縮後圖片顯示的質量為100,也就是壓縮率為0,不進行壓縮(作為compress的第二個參數)
提供了修改壓縮圖片格式和壓縮率以及修改緩存大小的set方法。同時該類還封裝了以下三個屬性
protected final File cacheDir;//緩存文件的保存Directory
protected final File reserveCacheDir;//後備緩存的Diectory,當cacheDir不存在的情況下就是用reserveCahceDir後備緩存
protected final FileNameGenerator fileNameGenerator;//文件名名稱生成器
構造函數
復制代碼
public BaseDiscCache(File cacheDir) {
this(cacheDir, null);
}
public BaseDiscCache(File cacheDir, File reserveCacheDir) {
this(cacheDir, reserveCacheDir, DefaultConfigurationFactory.createFileNameGenerator());
}
public BaseDiscCache(File cacheDir, File reserveCacheDir, FileNameGenerator fileNameGenerator) {
if (cacheDir == null) {
throw new IllegalArgumentException("cacheDir" + ERROR_ARG_NULL);
}
if (fileNameGenerator == null) {
throw new IllegalArgumentException("fileNameGenerator" + ERROR_ARG_NULL);
}
this.cacheDir = cacheDir;
this.reserveCacheDir = reserveCacheDir;
this.fileNameGenerator = fileNameGenerator;
}
復制代碼
1) 只有一個參數的構造函數只初始化了cacheDir,沒有用到後備緩存,且是以HashCodeFileNameGenerator來生成目標文件的文件名。
2) 兩個參數的構造器除了cacheDir和HashCodefileNameGenerator外,也可以初始化後備緩存
3) 三個參數的構造器要求必須初始化cacheDir並且必須初始化filenNameGenerator否則就報異常
get(String imageUri)
復制代碼
protected File getFile(String imageUri) {
String fileName = fileNameGenerator.generate(imageUri);
File dir = cacheDir;
if (!cacheDir.exists() && !cacheDir.mkdirs()) {
if (reserveCacheDir != null && (reserveCacheDir.exists() || reserveCacheDir.mkdirs())) {
dir = reserveCacheDir;
}
}
return new File(dir, fileName);
}
復制代碼
save(String imageUri, Bitmap bitmap)
復制代碼
public boolean save(String imageUri, Bitmap bitmap) throws IOException {
//獲取imageUri的File對象,該對象封裝了緩存路徑和圖片保存後的名稱
File imageFile = getFile(imageUri);
//獲取臨時保存文件的tmpFile對象
File tmpFile = new File(imageFile.getAbsolutePath() + TEMP_IMAGE_POSTFIX);
OutputStream os = new BufferedOutputStream(new FileOutputStream(tmpFile), bufferSize);
boolean savedSuccessfully = false;
try {
//調用compress把bitMap壓縮到tempFile中
savedSuccessfully = bitmap.compress(compressFormat, compressQuality, os);
} finally {
IoUtils.closeSilently(os);
//如果保存成功並且tempFile的文件沒有成功移動到imageFile的話,就刪除temFile
if (savedSuccessfully && !tmpFile.renameTo(imageFile)) {
savedSuccessfully = false;
}
if (!savedSuccessfully) {
tmpFile.delete();
}
}
//對bitmap進行垃圾回收
bitmap.recycle();
return savedSuccessfully;
}
復制代碼
BaseDiscCache有兩個擴展類,一個是不限制緩存大小的UnlimitedDiscCache和限制緩存時間的LimitedAgeDiscCache,其中UnlimitedDiscCache很簡單它只是簡單的繼承了BaseDiscCache並未對BaseDiscCache做任何擴展。
LimitedAgeDiscCache該類實現了在緩存中刪除被加載超過規定時間的文件:滿足以下條件的時候就從緩存中刪除文件:系統當前時間-文件的最新修改時間 > maxFileAge
LimitedAgeDiscCache
該類提供了兩個屬性:
1. maxFileAge(long)設置加載的超時的最大時間,改時間在構造器沖初始化,一經初始化就不能改變(設定文件存活的最長時間,當超過這個值,就刪除該文件)
2. loadingDates (Map<File,long>),該屬性是一個map類型的對象,key保存的要緩存的圖片文件,而value保存的是調用save方法是系統的當前時間,具體向loadingDates填充數據是在下面的rememberUsage方法中實現的,該方法在類中兩個save方法中調用,首先調用父類的save方法,然後在調用此方法
private void rememberUsage(String imageUri) {
File file = getFile(imageUri);
long currentTime = System.currentTimeMillis();
file.setLastModified(currentTime);
loadingDates.put(file, currentTime);
}
從緩存中獲取數據的方法為get(String imageUri)該類是重寫BaseDiscDache方法,該方法從loadingDates中獲取imageUri所代表的圖片的最新更新時間loadingDate,然後拿當前時間和loadingDate做差,如果差值大於maxFileAge也就是說查過了加載的最大時間,就刪除該imageUri所代表的file,並從loadingDates中的數據,當然如果map中沒有imageUri就不會涉及到超時的問題,此時就把image放入map中去,具體的實現如下
復制代碼
@Override
public File get(String imageUri) {
File file = super.get(imageUri);
if (file != null && file.exists()) {
boolean cached;
Long loadingDate = loadingDates.get(file);
if (loadingDate == null) {
cached = false;
loadingDate = file.lastModified();
} else {
cached = true;
}
if (System.currentTimeMillis() - loadingDate > maxFileAge) {
file.delete();
loadingDates.remove(file);
} else if (!cached) {
loadingDates.put(file, loadingDate);
}
}
return file;
}