簡單的ListView中item圖片異步加載

前言:android

    在android開發當中,從目標地址獲取圖片每每都是採用異步加載的方法。當徹底加載完圖片後在進行顯示,也有些是直接將加載的圖片一點一點的顯示出來。
這兩個區別只是對流的處理不一樣而已。如今就講講當圖片被徹底獲取到後在顯示的方法。
一,效果圖:
 
    初始化:                                                   獲取後:
                        
     1.1,效果思路:
           初始化的時候默認ImageView顯示一張白色的圖片,在加載完圖片之後,用心的圖片將白色圖片給替換掉,就達到效果了。
二,代碼實現:
     主要類介紹:
    AsyncImageLoadAdapter :ListView的自定義Adapter
    AsyncLoad :圖片異步加載類
    MainActivity :主Activity。
相關的佈局文件就不介紹了,在後面看就好了。
MainActivity:
 1 public class MainActivity extends Activity {
 2 
 3     private ListView listView;
 4 
 5     @Override
 6     protected void onCreate(Bundle savedInstanceState) {
 7         super.onCreate(savedInstanceState);
 8         setContentView(R.layout.activity_main);
 9         initView();
10         initData();
11     }
12 
13     private void initView() {
14         listView = (ListView) findView(R.id.listView);
15     }
16 
17     private void initData() {
18         List<HashMap<String, Object>> list = new ArrayList<HashMap<String, Object>>();
19         HashMap<String, Object> map1 = new HashMap<String, Object>();
20         map1.put("img",
21                 "http://img5.imgtn.bdimg.com/it/u=747474479,3247936386&fm=21&gp=0.jpg");
22         map1.put("txt", "條目一");
23         HashMap<String, Object> map2 = new HashMap<String, Object>();
24         map2.put("img",
25                 "http://pic.nipic.com/2007-12-22/2007122215556437_2.jpg");
26         map2.put("txt", "條目二");
27         HashMap<String, Object> map3 = new HashMap<String, Object>();
28         map3.put("img",
29                 "http://img1.imgtn.bdimg.com/it/u=1774561363,2410491846&fm=21&gp=0.jpg");
30         map3.put("txt", "條目三");
31         HashMap<String, Object> map4 = new HashMap<String, Object>();
32         map4.put("img",
33                 "http://img5.imgtn.bdimg.com/it/u=112049007,3368205326&fm=21&gp=0.jpg");
34         map4.put("txt", "條目四");
35         HashMap<String, Object> map5 = new HashMap<String, Object>();
36         map5.put("img",
37                 "http://img3.imgtn.bdimg.com/it/u=2953608063,4260501712&fm=21&gp=0.jpg");
38         map5.put("txt", "條目五");
39         list.add(map1);
40         list.add(map2);
41         list.add(map3);
42         list.add(map4);
43         list.add(map5);
44         AsyncImageLoadAdapter adapter = new AsyncImageLoadAdapter(
45                 MainActivity.this, list, listView);
46         listView.setAdapter(adapter);
47     }
48 
49     private <T> T findView(int id) {
50         return (T) findViewById(id);
51     }
52 
53 }
這個MainActivity挺簡單的,就是初始化相關控件,而後給ListView附上值而已。
 
 
AsyncImageLoadAdapter :
  
 1 public class AsyncImageLoadAdapter extends BaseAdapter {
 2 
 3     private Context context;
 4 
 5     private List<HashMap<String, Object>> listData;
 6 
 7     private AsyncLoad asyncLoad;
 8 
 9     private ListView listView;
10 
11     public AsyncImageLoadAdapter(Context context1,
12             List<HashMap<String, Object>> listData1, ListView listView1) {
13         this.context = context1;
14         this.listData = listData1;
15         this.asyncLoad = AsyncLoad.instance();
16         this.listView = listView1;
17     }
18 
19     @Override
20     public int getCount() {
21         return listData.size();
22     }
23 
24     @Override
25     public Object getItem(int position) {
26         return listData.get(position);
27     }
28 
29     @Override
30     public long getItemId(int position) {
31         return position;
32     }
33 
34     @Override
35     public View getView(int position, View convertView, ViewGroup parent) {
36 
37         if (convertView == null) {
38             convertView = LayoutInflater.from(context).inflate(
39                     R.layout.list_item, null);
40         }
41 
42         convertView.setTag(position);
43         ImageView imageView = (ImageView) convertView
44                 .findViewById(R.id.list_item_img);
45         imageView.setBackgroundResource(R.drawable.img_black);
46         TextView textView = (TextView) convertView
47                 .findViewById(R.id.list_item_tv);
48         textView.setText("我是條目" + position);
49         HashMap<String, Object> item = listData.get(position);
50         String imageStr = (String) item.get("img");
51         asyncLoad.loadImage(position, imageStr, new ILoadListener() {
52 
53             @SuppressLint("NewApi")
54             @Override
55             public void onSuccess(Integer pos, Drawable drawable) {
56                 View view = listView.findViewWithTag(pos);
57 
58                 if (view != null) {
59                     ImageView img = (ImageView) view
60                             .findViewById(R.id.list_item_img);
61 
62                     img.setBackground(drawable);
63                 }
64             }
65 
66             @Override
67             public void onFail(String failStr) {
68                 Toast.makeText(context, "錯誤信息是->" + failStr, Toast.LENGTH_SHORT)
69                         .show();
70             }
71         });
72 
73         return convertView;
74     }
75 }

這個Adapter中加載了一個list_item佈局(adapter中未用ViewHolder,不要去糾結這個,加上也可),來初始化未加載圖片的時候。主要要說的是這段代碼緩存

 1 asyncLoad.loadImage(position, imageStr, new ILoadListener() {
 2    @SuppressLint("NewApi")
 3    @Override
 4    public void onSuccess(Integer pos, Drawable drawable) {
 5     View view = listView.findViewWithTag(pos);
 6     if (view != null) {
 7      ImageView img = (ImageView) view
 8        .findViewById(R.id.list_item_img);
 9      img.setBackground(drawable);
10     }
11    }
12    @Override
13    public void onFail(String failStr) {
14     Toast.makeText(context, "錯誤信息是->" + failStr, Toast.LENGTH_SHORT)
15       .show();
16    }
17   });
這段代碼就進行了異步去加載圖片。
思路是:先將ListView中的每一個item設置一個Tag,便於後面找到對應的item中imageView進行圖片更新。
而後利用圖片加載類去網絡獲取圖片,獲取到圖片後採用接口回調的方式傳回圖片進行ImageView更新(UI線程中
執行的)
 
 
  AsyncLoad:
  
  1 public class AsyncLoad {
  2 
  3     private static AsyncLoad asyncLoad;
  4     /**
  5      * 圖片緩存容器,採用軟引用,當用戶手機內存不夠時候,系統自動回收其所佔有的內存
  6      */
  7     private HashMap<String, SoftReference<Drawable>> dataMap;
  8     /**
  9      * 用於在UI線程中進行UI更新
 10      */
 11     private Handler handler = new Handler();
 12 
 13     public synchronized static AsyncLoad instance() {
 14 
 15         if (asyncLoad == null)
 16             asyncLoad = new AsyncLoad();
 17         return asyncLoad;
 18 
 19     }
 20 
 21     public AsyncLoad() {
 22         dataMap = new HashMap<String, SoftReference<Drawable>>();
 23     }
 24 
 25     // 內部接口對圖片獲取的監聽
 26     interface ILoadListener {
 27         /**
 28          * 圖片加載成功後
 29          * 
 30          * @param pos
 31          *            對listView中pos位置
 32          * @param drawable
 33          *            網絡加載後獲取到的圖片
 34          */
 35         public void onSuccess(Integer pos, Drawable drawable);
 36 
 37         public void onFail(String failStr);
 38 
 39     }
 40 
 41     /**
 42      * 開啓線程對圖片進行加載
 43      * 
 44      * @param pos
 45      *            listView的pos位置進行加載
 46      * @param imageStr
 47      *            圖片地址
 48      * @param listener
 49      *            監聽接口
 50      */
 51 
 52     public void loadImage(final Integer pos, final String imageStr,
 53             final ILoadListener listener) {
 54 
 55         // 開啓一個線程獲取圖片資源
 56         new Thread(new Runnable() {
 57 
 58             @Override
 59             public void run() {
 60                 try {
 61                     Thread.sleep(5000);
 62                 } catch (InterruptedException e) {
 63                     e.printStackTrace();
 64                 }
 65                 load(pos, imageStr, listener);
 66             }
 67         }).start();
 68 
 69     }
 70 
 71     private void load(final Integer pos, final String imageStr,
 72             final ILoadListener listener) {
 73 
 74         if (dataMap.containsKey(imageStr)) {
 75 
 76             SoftReference<Drawable> softReference = dataMap.get(imageStr);
 77             final Drawable d = softReference.get();
 78             if (d != null) {
 79                 handler.post(new Runnable() {
 80 
 81                     @Override
 82                     public void run() {
 83 
 84                         listener.onSuccess(pos, d);
 85                     }
 86                 });
 87             }
 88             return;
 89 
 90         }
 91 
 92         try {
 93             final Drawable d = loadImgFromUrl(imageStr);
 94             if (d != null) {
 95                 SoftReference<Drawable> soft = new SoftReference<Drawable>(d);
 96                 dataMap.put(imageStr, soft);
 97             }
 98             handler.post(new Runnable() {
 99 
100                 @Override
101                 public void run() {
102 
103                     listener.onSuccess(pos, d);
104 
105                 }
106             });
107 
108         } catch (final IOException e) {
109             e.printStackTrace();
110             // 進行錯誤信息回報
111             handler.post(new Runnable() {
112 
113                 @Override
114                 public void run() {
115 
116                     listener.onFail(e.getMessage());
117 
118                 }
119             });
120 
121         }
122 
123     }
124 
125     /**
126      * 獲取網路圖片資源
127      * 
128      * @param url
129      * @return
130      * @throws IOException
131      */
132     private static Drawable loadImgFromUrl(String urlStr) throws IOException {
133         System.out.println("urlStr->" + urlStr);
134         URL url;
135         InputStream i = null;
136         url = new URL(urlStr);
137         i = (InputStream) url.getContent();
138         Drawable drawable = Drawable.createFromStream(i, "img");
139 
140         return drawable;
141 
142     }
143 
144 }
要想更新界面上內容,就必須在UI線程中去更新,因此就採用handler.post方法,post裏面Runnable中
run執行的代碼實際上是在UI線程中執行的,因此就符合android的規矩,才能進行更新
 
 
 
參考文章:http://blog.csdn.net/harvic880925/article/details/17766027#t7
源碼下載: 下載
相關文章
相關標籤/搜索