昨天參加一個面試,面試官讓當場寫一個相似於新聞列表的頁面,文本數據和圖片都從網絡上獲取,想起我還沒寫過ListView異步加載圖片並實現圖文混排效果的文章,so,今天就來寫一下,介紹一下經驗。java
ListView加載文本數據都是很簡單的,即便是異步獲取文本數據。可是異步加載圖片就稍微有一點麻煩,既要得到一個比較好的用戶體驗,還要防止出現圖片錯位等各類不良BUG,其實要考慮的東西仍是挺多的。好了,咱們先來看一下咱們今天要實現的一個效果圖:android
看起來彷佛並不難,確實,咱們今天的核心問題只有一個,就是怎麼異步加載圖片,而且沒有違和感。git
好了,廢話很少說,先來看主佈局文件:github
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context="com.example.listview.MainActivity" > <ListView android:id="@+id/lv" android:layout_width="match_parent" android:layout_height="match_parent" > </ListView> </RelativeLayout>
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="100dp" > <ImageView android:id="@+id/iv" android:layout_width="80dp" android:layout_height="90dp" android:layout_centerVertical="true" android:padding="5dp" android:src="@drawable/ic_launcher" /> <TextView android:id="@+id/title" android:layout_width="match_parent" android:layout_height="wrap_content" android:layout_marginTop="20dp" android:layout_toRightOf="@id/iv" android:gravity="center_vertical" android:text="人社部:養老轉移已有初稿" android:textSize="14sp" android:textStyle="bold" /> <TextView android:id="@+id/summary" android:layout_width="match_parent" android:layout_height="wrap_content" android:layout_below="@id/title" android:layout_marginTop="8dp" android:layout_toRightOf="@id/iv" android:text="人社部:養老轉移已有初稿" android:textSize="12sp" /> </RelativeLayout>
public class MainActivity extends Activity { private ListView lv; private List<News> list; private String HTTPURL = "http://litchiapi.jstv.com/api/GetFeeds?column=3&PageSize=20&pageIndex=1&val=100511D3BE5301280E0992C73A9DEC41"; private Handler mHandler = new Handler(){ @Override public void handleMessage(Message msg) { super.handleMessage(msg); switch (msg.what) { case 0: MyAdaper adapter = new MyAdaper(list); lv.setAdapter(adapter); break; default: break; } } }; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); lv = (ListView) this.findViewById(R.id.lv); initData(); } private void initData() { list = new ArrayList<News>(); OkHttpClient client = new OkHttpClient(); Request request = new Request.Builder().url(HTTPURL).build(); Call call = client.newCall(request); call.enqueue(new Callback() { @Override public void onResponse(Response response) throws IOException { try { JSONObject jo1 = new JSONObject(response.body().string()); JSONObject jo2 = jo1.getJSONObject("paramz"); JSONArray ja = jo2.getJSONArray("feeds"); News news = null; for (int i = 0; i < ja.length(); i++) { JSONObject data = ja.getJSONObject(i).getJSONObject( "data"); String imageUrl = "http://litchiapi.jstv.com" + data.getString("cover"); String title = data.getString("subject"); String summary = data.getString("summary"); news = new News(imageUrl, title, summary); list.add(news); } } catch (JSONException e) { e.printStackTrace(); } mHandler.obtainMessage(0).sendToTarget(); } @Override public void onFailure(Request arg0, IOException arg1) { } }); } }
public class News { private String imageUrl; private String title; private String summary; public String getImageUrl() { return imageUrl; } public void setImageUrl(String imageUrl) { this.imageUrl = imageUrl; } public String getTitle() { return title; } public void setTitle(String title) { this.title = title; } public String getSummary() { return summary; } public void setSummary(String summary) { this.summary = summary; } public News(String imageUrl, String title, String summary) { this.imageUrl = imageUrl; this.title = title; this.summary = summary; } public News() { } }
private LruCache<String, BitmapDrawable> mImageCache;
int maxCache = (int) Runtime.getRuntime().maxMemory(); int cacheSize = maxCache / 8; mImageCache = new LruCache<String, BitmapDrawable>(cacheSize) { @Override protected int sizeOf(String key, BitmapDrawable value) { return value.getBitmap().getByteCount(); } };
mImageCache.get(key)
mImageCache.put(key, bitmapDrawable);
public class MyAdaper extends BaseAdapter { private List<News> list; private ListView listview; private LruCache<String, BitmapDrawable> mImageCache; public MyAdaper(List<News> list) { super(); this.list = list; int maxCache = (int) Runtime.getRuntime().maxMemory(); int cacheSize = maxCache / 8; mImageCache = new LruCache<String, BitmapDrawable>(cacheSize) { @Override protected int sizeOf(String key, BitmapDrawable value) { return value.getBitmap().getByteCount(); } }; } @Override public int getCount() { return list.size(); } @Override public Object getItem(int position) { return list.get(position); } @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View convertView, ViewGroup parent) { if (listview == null) { listview = (ListView) parent; } ViewHolder holder = null; if (convertView == null) { convertView = LayoutInflater.from(parent.getContext()).inflate( R.layout.listview_item, null); holder = new ViewHolder(); holder.iv = (ImageView) convertView.findViewById(R.id.iv); holder.title = (TextView) convertView.findViewById(R.id.title); holder.summary = (TextView) convertView.findViewById(R.id.summary); convertView.setTag(holder); } else { holder = (ViewHolder) convertView.getTag(); } News news = list.get(position); holder.title.setText(news.getTitle()); holder.summary.setText(news.getSummary()); holder.iv.setTag(news.getImageUrl()); // 若是本地已有緩存,就從本地讀取,不然從網絡請求數據 if (mImageCache.get(news.getImageUrl()) != null) { holder.iv.setImageDrawable(mImageCache.get(news.getImageUrl())); } else { ImageTask it = new ImageTask(); it.execute(news.getImageUrl()); } return convertView; } class ViewHolder { ImageView iv; TextView title, summary; } class ImageTask extends AsyncTask<String, Void, BitmapDrawable> { private String imageUrl; @Override protected BitmapDrawable doInBackground(String... params) { imageUrl = params[0]; Bitmap bitmap = downloadImage(); BitmapDrawable db = new BitmapDrawable(listview.getResources(), bitmap); // 若是本地還沒緩存該圖片,就緩存 if (mImageCache.get(imageUrl) == null) { mImageCache.put(imageUrl, db); } return db; } @Override protected void onPostExecute(BitmapDrawable result) { // 經過Tag找到咱們須要的ImageView,若是該ImageView所在的item已被移出頁面,就會直接返回null ImageView iv = (ImageView) listview.findViewWithTag(imageUrl); if (iv != null && result != null) { iv.setImageDrawable(result); } } /** * 根據url從網絡上下載圖片 * * @return */ private Bitmap downloadImage() { HttpURLConnection con = null; Bitmap bitmap = null; try { URL url = new URL(imageUrl); con = (HttpURLConnection) url.openConnection(); con.setConnectTimeout(5 * 1000); con.setReadTimeout(10 * 1000); bitmap = BitmapFactory.decodeStream(con.getInputStream()); } catch (MalformedURLException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } finally { if (con != null) { con.disconnect(); } } return bitmap; } } }好了,listview圖文混排就說到這裏,有問題歡迎留言討論。