android非同步載入圖片並緩衝到本地實現方法

來源:互聯網
上載者:User

在android項目中訪問網狀圖片是非常普遍性的事情,如果我們每次請求都要訪問網路來擷取圖片,會非常耗費流量,而且圖片佔用記憶體空間也比較大,圖片過多且不釋放的話很容易造成記憶體溢出。針對上面遇到的兩個問題,首先耗費流量我們可以將圖片第一次載入上面緩衝到本地,以後如果本地有就直接從本地載入。圖片過多造成記憶體溢出,這個是最不容易解決的,要想一些好的緩衝策略,比如大圖片使用LRU緩衝策略或懶載入緩衝策略。今天首先介紹一下本機快取圖片。
首先看一下非同步載入緩衝本地代碼: 複製代碼 代碼如下:publicclassAsyncBitmapLoader
{
/**
*記憶體配置圖片軟引用緩衝
*/
privateHashMap<String,SoftReference<Bitmap>>imageCache=null;
publicAsyncBitmapLoader()
{
imageCache=newHashMap<String,SoftReference<Bitmap>>();
}
publicBitmaploadBitmap(finalImageViewimageView,finalStringimageURL,finalImageCallBackimageCallBack)
{
//在記憶體緩衝中,則返回Bitmap對象
if(imageCache.containsKey(imageURL))
{
SoftReference<Bitmap>reference=imageCache.get(imageURL);
Bitmapbitmap=reference.get();
if(bitmap!=null)
{
returnbitmap;
}
}
else
{
/**
*加上一個對本機快取的尋找
*/
StringbitmapName=imageURL.substring(imageURL.lastIndexOf("/")+1);
FilecacheDir=newFile("/mnt/sdcard/test/");
File[]cacheFiles=cacheDir.listFiles();
inti=0;
if(null!=cacheFiles){
for(;i<cacheFiles.length;i++)
{
if(bitmapName.equals(cacheFiles[i].getName()))
{
break;
}
}
if(i<cacheFiles.length)
{
returnBitmapFactory.decodeFile("/mnt/sdcard/test/"+bitmapName);
}
}
}
finalHandlerhandler=newHandler()
{
/*(non-Javadoc)
*@seeandroid.os.Handler#handleMessage(android.os.Message)
*/
@Override
publicvoidhandleMessage(Messagemsg)
{
//TODOAuto-generatedmethodstub
imageCallBack.imageLoad(imageView,(Bitmap)msg.obj);
}
};
//如果不在記憶體緩衝中,也不在本地(被jvm回收掉),則開啟線程下載圖片
newThread()
{
/*(non-Javadoc)
*@seejava.lang.Thread#run()
*/
@Override
publicvoidrun()
{
//TODOAuto-generatedmethodstub
InputStreambitmapIs=HttpUtils.getStreamFromURL(imageURL);
Bitmapbitmap=BitmapFactory.decodeStream(bitmapIs);
imageCache.put(imageURL,newSoftReference<Bitmap>(bitmap));
Messagemsg=handler.obtainMessage(0,bitmap);
handler.sendMessage(msg);
Filedir=newFile("/mnt/sdcard/test/");
if(!dir.exists())
{
dir.mkdirs();
}
FilebitmapFile=newFile("/mnt/sdcard/test/"+
imageURL.substring(imageURL.lastIndexOf("/")+1));
if(!bitmapFile.exists())
{
try
{
bitmapFile.createNewFile();
}
catch(IOExceptione)
{
//TODOAuto-generatedcatchblock
e.printStackTrace();
}
}
FileOutputStreamfos;
try
{
fos=newFileOutputStream(bitmapFile);
bitmap.compress(Bitmap.CompressFormat.PNG,
100,fos);
fos.close();
}
catch(FileNotFoundExceptione)
{
//TODOAuto-generatedcatchblock
e.printStackTrace();
}
catch(IOExceptione)
{
//TODOAuto-generatedcatchblock
e.printStackTrace();
}
}
}.start();
returnnull;
}
publicinterfaceImageCallBack
{
publicvoidimageLoad(ImageViewimageView,Bitmapbitmap);
}
}

這是一個封裝好的非同步載入圖片類,緩衝了兩份,一份是使用軟引用緩衝到記憶體中,一份是緩衝到本地sd卡,如果記憶體中沒有,則從本地尋找,如果本地沒有則從網路擷取圖片。 複製代碼 代碼如下:publicclassHttpUtils{
publicstaticInputStreamgetStreamFromURL(StringimageURL){
InputStreamin=null;
try{
URLurl=newURL(imageURL);
HttpURLConnectionconnection=(HttpURLConnection)url.openConnection();
in=connection.getInputStream();

}catch(Exceptione){
//TODOAuto-generatedcatchblock
e.printStackTrace();
}
returnin;

}
}

這是一個訪問網路擷取類。不細說了。
下面看一下如何使用封裝好的非同步載入圖片的類: 複製代碼 代碼如下:publicclassImageCacheActivityextendsActivity{
/**Calledwhentheactivityisfirstcreated.*/
privateListViewlistview;
@Override
publicvoidonCreate(BundlesavedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
listview=(ListView)findViewById(R.id.listView_list);
MyAdapteradapter=newMyAdapter();
listview.setAdapter(adapter);
}
privateclassMyAdapterextendsBaseAdapter{
privateAsyncBitmapLoaderasyncBitmapLoader;
publicMyAdapter(){
asyncBitmapLoader=newAsyncBitmapLoader();
}
@Override
publicintgetCount(){
//TODOAuto-generatedmethodstub
return10;
}
@Override
publicObjectgetItem(intposition){
//TODOAuto-generatedmethodstub
returnnull;
}
@Override
publiclonggetItemId(intposition){
//TODOAuto-generatedmethodstub
return0;
}
@Override
publicViewgetView(intposition,ViewconvertView,ViewGroupparent){
//TODOAuto-generatedmethodstub
if(convertView==null){
convertView=LayoutInflater.from(getApplicationContext()).inflate(R.layout.list_item,null);
}
ImageViewimage=(ImageView)convertView.findViewById(R.id.addexam_list_icon);
StringimageURL="http://s.ata.net.cn/4f98db46908987a21a000003/logo/2012/04/114_80aaf295c083d07a496743699aac3193.png";
Bitmapbitmap=asyncBitmapLoader.loadBitmap(image,imageURL,newImageCallBack(){

@Override
publicvoidimageLoad(ImageViewimageView,Bitmapbitmap){
//TODOAuto-generatedmethodstub
imageView.setImageBitmap(bitmap);
}
});
if(bitmap==null)
{
image.setImageResource(R.drawable.ic_launcher);
}
else
{
image.setImageBitmap(bitmap);
}

returnconvertView;
}

}
}

這樣就完成了,網路擷取不到bitmap則顯示預設圖片。

這是一種很實用的方法,大家自己試試吧!

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.