前言
我們經常遇到從網絡獲取圖片,爲了使圖片查看流暢,我們肯定要使用緩存,大部分我們會使用內存緩存,但是android內存緩存畢竟是有限的,這樣的話,我們必須使用文件來緩存部分圖片。
思路
當我們把一張圖片從網絡下載成功以後,這個圖片會被加入內存緩存和文件緩存,內存緩存來說請參考Android內存溢出大總結,對於文件緩存來說,這張圖片將被以url的哈希值加cach後綴名的形式存儲在SD卡上,這樣,當下一次再需要同一個url的圖片的時候,就不需要從網絡下載了,而是直接通過url來進行查找。同時一張圖片被訪問時,它的最後修改時間將被更新,這樣的意義在於:當SD卡空間不足的時候,將會按照最後修改時間來刪除40%緩存的圖片,確切來說,那些修改時間比較早的圖片將會被刪除。
代碼
package com.ty.highway.highwaysystem.support.utils.cache;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.Environment;
import android.os.StatFs;
import android.util.Log;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.util.Arrays;
import java.util.Comparator;
/**
* Created by fuweiwei on 2016/1/26.
* 文件緩存
* 內存緩存是有限的,其它的文件緩存,當文件緩存操作一定量時我們刪除之前的緩存
*/
public class ImageFileCache
{
private static final String TAG = "ImageFileCache";
//圖片緩存目錄
private static final String IMGCACHDIR = "/sdcard/ImgCach";
//保存的cache文件寬展名
private static final String CACHETAIL = ".cach";
private static final int MB = 1024*1024;
private static final int CACHE_SIZE = 1;
//當SD卡剩餘空間小於10M的時候會清理緩存
private static final int FREE_SD_SPACE_NEEDED_TO_CACHE = 10;
public ImageFileCache()
{
//清理部分文件緩存
removeCache(IMGCACHDIR);
}
/**
* 從緩存中獲取圖片
*/
public Bitmap getImageFromFile(final String url)
{
final String path = IMGCACHDIR + "/" + convertUrlToFileName(url);
File file = new File(path);
if (file != null && file.exists())
{
Bitmap bmp = BitmapFactory.decodeFile(path);
if (bmp == null)
{
file.delete();
}
else
{
updateFileTime(path);
Log.d(TAG, "get bmp from FileCache,url=" + url);
return bmp;
}
}
return null;
}
/**
* 將圖片存入文件緩存
*/
public void saveBitmapToFile(Bitmap bm, String url)
{
if (bm == null) {
return;
}
//判斷sdcard上的空間
if (FREE_SD_SPACE_NEEDED_TO_CACHE > SdCardFreeSpace())
{
//SD空間不足
return;
}
String filename = convertUrlToFileName(url);
File dirFile = new File(IMGCACHDIR);
if (!dirFile.exists())
dirFile.mkdirs();
File file = new File(IMGCACHDIR +"/" + filename);
try
{
file.createNewFile();
OutputStream outStream = new FileOutputStream(file);
bm.compress(Bitmap.CompressFormat.JPEG, 100, outStream);
outStream.flush();
outStream.close();
}
catch (FileNotFoundException e)
{
Log.d(TAG, "FileNotFoundException");
}
catch (IOException e)
{
Log.d(TAG, "IOException");
}
}
/**
* 計算存儲目錄下的文件大小,
* 當文件總大小大於規定的CACHE_SIZE或者sdcard剩餘空間小於FREE_SD_SPACE_NEEDED_TO_CACHE的規定
* 那麼刪除40%最近沒有被使用的文件
*/
private boolean removeCache(String dirPath)
{
File dir = new File(dirPath);
File[] files = dir.listFiles();
if (files == null)
{
return true;
}
if (!android.os.Environment.getExternalStorageState().equals(android.os.Environment.MEDIA_MOUNTED))
{
return false;
}
int dirSize = 0;
for (int i = 0; i < files.length; i++)
{
if (files[i].getName().contains(CACHETAIL))
{
dirSize += files[i].length();
}
}
if (dirSize > CACHE_SIZE * MB || FREE_SD_SPACE_NEEDED_TO_CACHE > SdCardFreeSpace())
{
int removeFactor = (int) (0.4 * files.length);
Arrays.sort(files, new FileLastModifSort());
for (int i = 0; i < removeFactor; i++)
{
if (files[i].getName().contains(CACHETAIL))
{
files[i].delete();
}
}
}
if (SdCardFreeSpace() <= CACHE_SIZE)
{
return false;
}
return true;
}
/**
* 修改文件的最後修改時間
*/
public void updateFileTime(String path)
{
File file = new File(path);
long newModifiedTime = System.currentTimeMillis();
file.setLastModified(newModifiedTime);
}
/**
* 計算SD卡上的剩餘空間
*/
private int SdCardFreeSpace()
{
StatFs stat = new StatFs(Environment.getExternalStorageDirectory().getPath());
double sdFreeMB = ((double)stat.getAvailableBlocks() * (double) stat.getBlockSize()) / MB;
return (int) sdFreeMB;
}
/**
* 將url轉成文件名
*/
private String convertUrlToFileName(String url)
{
return url.hashCode() + CACHETAIL;
}
/**
* 根據文件的最後修改時間進行排序
*/
private class FileLastModifSort implements Comparator<File>
{
public int compare(File file0, File file1)
{
if (file0.lastModified() > file1.lastModified())
{
return 1;
}
else if (file0.lastModified() == file1.lastModified())
{
return 0;
}
else
{
return -1;
}
}
}
}