Android開發(fā)之ImageLoader本地緩存
ImageLoader是一個圖片緩存的開源庫,提供了強大的圖片緩存機(jī)制,很多開發(fā)者都在使用,今天給大家介紹Android開發(fā)之ImageLoader本地緩存,具體內(nèi)容如下所示:
本地緩存在緩存文件時對文件名稱的修改提供了兩種方式,每一種方式對應(yīng)了一個Java類
1) HashCodeFileNameGenerator ,該類負(fù)責(zé)獲取文件名稱的hashcode然后轉(zhuǎn)換成字符串。
2) Md5FileNameGenerator ,該類把源文件的名稱同過md5加密后保存。
兩個類都繼承了FileNameGenerator接口
在DefaultConfigurationFactory類中提供了一個工廠方法createFileNameGenerator,該方法返回了一個默認(rèn)的FileNameGenerator對象:HashCodeFileNameGenerator.
public static FileNameGenerator createFileNameGenerator() { return new HashCodeFileNameGenerator(); }
實現(xiàn)
首先定義了DiscCacheAware接口,該接口提供了如下方法
File getFileDectory() 返回磁盤緩存的根目錄 File get(String imageUri) 根據(jù)uri從緩存中獲取圖片 boolean save(String imageUri,InputStream iamgeStream,IoUtils.CopyListener listener) 把圖片保存在磁盤緩存上 boolean save(String imageUri,Bitmap bitmap) 保存bitmap對象到磁盤緩存上 boolean remove(imageUri) 根據(jù)imageUri刪除文件 void close() 關(guān)閉磁盤緩存,釋放資源 void clear() 清空磁盤緩存
然后定義了另外一個沒方法的接口DiskCache,該接口只是簡單的繼承了DiscCacheAware接口。
BaseDiscCache實現(xiàn)了DiskCache,該類是個抽象類,該類定義了磁盤緩沖區(qū)的以下的屬性:
1) 默認(rèn)的緩存大小為32k
2) 默認(rèn)壓縮后的圖片格式為PNG(作為Bitmap的compress方法的第一個參數(shù))
3) 默認(rèn)壓縮后圖片顯示的質(zhì)量為100,也就是壓縮率為0,不進(jìn)行壓縮(作為compress的第二個參數(shù))
提供了修改壓縮圖片格式和壓縮率以及修改緩存大小的set方法。同時該類還封裝了以下三個屬性
protected final File cacheDir;//緩存文件的保存Directory protected final File reserveCacheDir;//后備緩存的Diectory,當(dāng)cacheDir不存在的情況下就是用reserveCahceDir后備緩存 protected final FileNameGenerator fileNameGenerator;//文件名名稱生成器
構(gòu)造函數(shù)
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) 只有一個參數(shù)的構(gòu)造函數(shù)只初始化了cacheDir,沒有用到后備緩存,且是以HashCodeFileNameGenerator來生成目標(biāo)文件的文件名。
2) 兩個參數(shù)的構(gòu)造器除了cacheDir和HashCodefileNameGenerator外,也可以初始化后備緩存
3) 三個參數(shù)的構(gòu)造器要求必須初始化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 { //調(diào)用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進(jìn)行垃圾回收 bitmap.recycle(); return savedSuccessfully; }
BaseDiscCache有兩個擴(kuò)展類,一個是 不限制緩存大小的 UnlimitedDiscCache 和 限制緩存時間的LimitedAgeDiscCache, 其中UnlimitedDiscCache很簡單它只是簡單的繼承了BaseDiscCache并未對BaseDiscCache做任何擴(kuò)展。
LimitedAgeDiscCache 該類實現(xiàn)了在緩存中刪除被加載超過規(guī)定時間的文件: 滿足以下條件的時候就從緩存中刪除文件:系統(tǒng)當(dāng)前時間-文件的最新修改時間 > maxFileAge
LimitedAgeDiscCache
該類提供了兩個屬性:
1. maxFileAge(long)設(shè)置加載的超時的最大時間,改時間在構(gòu)造器沖初始化,一經(jīng)初始化就不能改變(設(shè)定文件存活的最長時間,當(dāng)超過這個值,就刪除該文件)
2. loadingDates (Map<File,long>),該屬性是一個map類型的對象,key保存的要緩存的圖片文件,而value保存的是調(diào)用save方法是系統(tǒng)的當(dāng)前時間,具體向loadingDates填充數(shù)據(jù)是在下面的rememberUsage方法中實現(xiàn)的,該方法在類中兩個save方法中調(diào)用,首先調(diào)用父類的save方法,然后在調(diào)用此方法
private void rememberUsage(String imageUri) { File file = getFile(imageUri); long currentTime = System.currentTimeMillis(); file.setLastModified(currentTime); loadingDates.put(file, currentTime); }
從緩存中獲取數(shù)據(jù)的方法為get(String imageUri)該類是重寫B(tài)aseDiscDache方法,該方法從loadingDates中獲取imageUri所代表的圖片的最新更新時間loadingDate, 然后拿當(dāng)前時間和loadingDate做差,如果差值大于maxFileAge也就是說查過了加載的最大時間,就刪除該imageUri所代表的file,并從loadingDates中的數(shù)據(jù),當(dāng)然如果map中沒有imageUri就不會涉及到超時的問題,此時就把image放入map中去 ,具體的實現(xiàn)如下
@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; }
- Android ListView實現(xiàn)ImageLoader圖片加載的方法
- Android Universal ImageLoader 緩存圖片
- Android Imageloader的配置的實現(xiàn)代碼
- Android ImageLoader第三方框架解析
- Android開發(fā)之ImageLoader使用詳解
- Android圖片加載的緩存類
- 非常實用的Android圖片工具類
- Android開發(fā)之多媒體文件獲取工具類實例【音頻,視頻,圖片等】
- Android開發(fā)之超強圖片工具類BitmapUtil完整實例
- Android開發(fā)之圖片壓縮工具類完整實例
- Android編程圖片加載類ImageLoader定義與用法實例分析
相關(guān)文章
Android開發(fā)之TimePicker控件用法實例詳解
這篇文章主要介紹了Android開發(fā)之TimePicker控件用法,結(jié)合實例形式詳細(xì)分析了Android項目的建立及TimePicker控件的具體使用技巧,需要的朋友可以參考下2016-02-02Android常用三方庫混淆規(guī)則整理(小結(jié))
這篇文章主要介紹了Android常用三方庫混淆規(guī)則整理(小結(jié)),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-07-07Android 使用Vitamio打造自己的萬能播放器(1)——準(zhǔn)備
本文主要介紹Android Vitamio,在Android開發(fā)視頻播放器的時候,大家經(jīng)常會遇到系統(tǒng)版本和不同的Android手機(jī)不同導(dǎo)致開發(fā)的軟件不能完美適用,這里給大家介紹個播放器插件可以適應(yīng)所有Android設(shè)備2016-07-07Android編程之手機(jī)壁紙WallPaper設(shè)置方法示例
這篇文章主要介紹了Android編程之手機(jī)壁紙WallPaper設(shè)置方法,結(jié)合實例形式分析了Android手機(jī)壁紙WallPaper的相關(guān)設(shè)置與使用技巧,需要的朋友可以參考下2017-08-08Android網(wǎng)絡(luò)開發(fā)中GET與POST請求詳解
這篇文章主要介紹了android實現(xiàn)網(wǎng)絡(luò)請求的get和post請求的簡單封裝與使用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧<BR>2022-12-12