Android — ImageLoader本地緩存

地緩存                                                                              

 

在緩存文件時對文件名稱的修改提供瞭兩種方式,每一種方式對應瞭一個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;  

    }

發佈留言

發佈留言必須填寫的電子郵件地址不會公開。