在android项目中访问网络图片是非常普遍性的事情,如果我们每次请求都要访问网络来获取图片,会非常耗费流量,而且图片占用内存空间也比较大,图片过多且不释放的话很容易造成内存溢出。针对上面遇到的两个问题,首先耗费流量我们可以将图片第一次加载上面缓存到本地,以后如果本地有就直接从本地加载。图片过多造成内存溢出,这个是最不容易解决的,要想一些好的缓存策略,比如大图片使用lru缓存策略或懒加载缓存策略,首先介绍一下本地缓存图片。
首先看一下异步加载缓存本地代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
|
public class asyncbitmaploader
{
/**
* 内存图片软引用缓冲
*/
private hashmap<string, softreference<bitmap>> imagecache = null ;
public asyncbitmaploader()
{
imagecache = new hashmap<string, softreference<bitmap>>();
}
public bitmap loadbitmap( final imageview imageview, final string imageurl, final imagecallback imagecallback)
{
//在内存缓存中,则返回bitmap对象
if (imagecache.containskey(imageurl))
{
softreference<bitmap> reference = imagecache.get(imageurl);
bitmap bitmap = reference.get();
if (bitmap != null )
{
return bitmap;
}
}
else
{
/**
* 加上一个对本地缓存的查找
*/
string bitmapname = imageurl.substring(imageurl.lastindexof( "/" ) + 1 );
file cachedir = new file( "/mnt/sdcard/test/" );
file[] cachefiles = cachedir.listfiles();
int i = 0 ;
if ( null !=cachefiles){
for (; i<cachefiles.length; i++)
{
if (bitmapname.equals(cachefiles[i].getname()))
{
break ;
}
}
if (i < cachefiles.length)
{
return bitmapfactory.decodefile( "/mnt/sdcard/test/" + bitmapname);
}
}
}
final handler handler = new handler()
{
/* (non-javadoc)
* @see android.os.handler#handlemessage(android.os.message)
*/
@override
public void handlemessage(message msg)
{
// todo auto-generated method stub
imagecallback.imageload(imageview, (bitmap)msg.obj);
}
};
//如果不在内存缓存中,也不在本地(被jvm回收掉),则开启线程下载图片
new thread()
{
/* (non-javadoc)
* @see java.lang.thread#run()
*/
@override
public void run()
{
// todo auto-generated method stub
inputstream bitmapis = httputils.getstreamfromurl(imageurl);
bitmap bitmap = bitmapfactory.decodestream(bitmapis);
imagecache.put(imageurl, new softreference<bitmap>(bitmap));
message msg = handler.obtainmessage( 0 , bitmap);
handler.sendmessage(msg);
file dir = new file( "/mnt/sdcard/test/" );
if (!dir.exists())
{
dir.mkdirs();
}
file bitmapfile = new file( "/mnt/sdcard/test/" +
imageurl.substring(imageurl.lastindexof( "/" ) + 1 ));
if (!bitmapfile.exists())
{
try
{
bitmapfile.createnewfile();
}
catch (ioexception e)
{
// todo auto-generated catch block
e.printstacktrace();
}
}
fileoutputstream fos;
try
{
fos = new fileoutputstream(bitmapfile);
bitmap.compress(bitmap.compressformat.png,
100 , fos);
fos.close();
}
catch (filenotfoundexception e)
{
// todo auto-generated catch block
e.printstacktrace();
}
catch (ioexception e)
{
// todo auto-generated catch block
e.printstacktrace();
}
}
}.start();
return null ;
}
public interface imagecallback
{
public void imageload(imageview imageview, bitmap bitmap);
}
}
|
这是一个封装好的异步加载图片类,缓存了两份,一份是使用软引用缓存到内存中,一份是缓存到本地sd卡,如果内存中没有,则从本地查找,如果本地没有则从网络获取图片。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
public class httputils {
public static inputstream getstreamfromurl(string imageurl) {
inputstream in= null ;
try {
url url= new url(imageurl);
httpurlconnection connection=(httpurlconnection) url.openconnection();
in=connection.getinputstream();
} catch (exception e) {
// todo auto-generated catch block
e.printstacktrace();
}
return in;
}
}
|
这是一个访问网络获取类,不细说了。
下面看一下如何使用封装好的异步加载图片的类:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
|
public class imagecacheactivity extends activity {
/** called when the activity is first created. */
private listview listview;
@override
public void oncreate(bundle savedinstancestate) {
super .oncreate(savedinstancestate);
setcontentview(r.layout.main);
listview=(listview) findviewbyid(r.id.listview_list);
myadapter adapter= new myadapter();
listview.setadapter(adapter);
}
private class myadapter extends baseadapter{
private asyncbitmaploader asyncbitmaploader;
public myadapter(){
asyncbitmaploader= new asyncbitmaploader();
}
@override
public int getcount() {
// todo auto-generated method stub
return 10 ;
}
@override
public object getitem( int position) {
// todo auto-generated method stub
return null ;
}
@override
public long getitemid( int position) {
// todo auto-generated method stub
return 0 ;
}
@override
public view getview( int position, view convertview, viewgroup parent) {
// todo auto-generated method stub
if (convertview== null ){
convertview=layoutinflater.from(getapplicationcontext()).inflate(r.layout.list_item, null );
}
imageview image=(imageview) convertview.findviewbyid(r.id.addexam_list_icon);
string imageurl= "http://s.ata.net.cn/4f98db46908987a21a000003/logo/2012/04/114_80aaf295c083d07a496743699aac3193.png" ;
bitmap bitmap=asyncbitmaploader.loadbitmap(image, imageurl, new imagecallback() {
@override
public void imageload(imageview imageview, bitmap bitmap) {
// todo auto-generated method stub
imageview.setimagebitmap(bitmap);
}
});
if (bitmap == null )
{
image.setimageresource(r.drawable.ic_launcher);
}
else
{
image.setimagebitmap(bitmap);
}
return convertview;
}
}
}
|
这样就完成了,网络获取不到bitmap则显示默认图片。
这是一种很实用的方法,大家自己试试吧!
希望本文所述对大家学习android软件编程有所帮助。