首页 > 系统 > Android > 正文

Android编程学习之异步加载图片的方法

2019-10-24 20:35:49
字体:
来源:转载
供稿:网友

这篇文章主要介绍了Android编程学习之异步加载图片的方法,以实例形式较为详细的分析了Android异步加载图片所涉及的页面布局及功能实现技巧,具有一定参考借鉴价值,需要的朋友可以参考下

本文实例讲述了Android编程学习之异步加载图片的方法。分享给大家供大家参考,具体如下:

最近在android开发中碰到比较棘手的问题,就是加载图片内存溢出。我开发的是一个新闻应用,应用中用到大量的图片,一个界面中可能会有上百张图片。开发android应用的朋友可能或多或少碰到加载图片内存溢出问题,一般情况下,加载一张大图就会导致内存溢出,同样,加载多张图片内存溢出的概率也很高。

列一下网络上查到的一般做法:

1.使用BitmapFactory.Options对图片进行压缩

2.优化加载图片的adapter中的getView方法,使之尽可能少占用内存

3.使用异步加载图片的方式,使图片在页面加载后慢慢载入进来。

1、2步骤是必须做足的工作,但是对于大量图片的列表仍然无法解决内存溢出的问题,采用异步加载图片的方式才能有效解决图片加载内存溢出问题。

测试的效果图如下:

Android编程学习之异步加载图片的方法

在这里我把主要的代码贴出来,给大家分享一下。

1、首先是MainActivity和activity_main.xml布局文件的代码。

(1)、MainActivity的代码如下:

 

 
  1. package net.loonggg.test;  
  2. import java.util.List;  
  3. import net.loonggg.adapter.MyAdapter;  
  4. import net.loonggg.bean.Menu;  
  5. import net.loonggg.util.HttpUtil;  
  6. import net.loonggg.util.Utils;  
  7. import android.app.Activity;  
  8. import android.app.ProgressDialog;  
  9. import android.os.AsyncTask;  
  10. import android.os.Bundle;  
  11. import android.view.Window;  
  12. import android.widget.ListView;  
  13. public class MainActivity extends Activity {  
  14. private ListView lv;  
  15. private MyAdapter adapter;  
  16. private ProgressDialog pd;  
  17. @Override 
  18. protected void onCreate(Bundle savedInstanceState) {  
  19. requestWindowFeature(Window.FEATURE_NO_TITLE);  
  20. super.onCreate(savedInstanceState);  
  21. setContentView(R.layout.activity_main);  
  22. lv = (ListView) findViewById(R.id.lv);  
  23. pd = new ProgressDialog(this);  
  24. pd.setTitle("加载菜单");  
  25. pd.setMessage("正在加载");  
  26. adapter = new MyAdapter(this);  
  27. new MyTask().execute("1");  
  28. }  
  29. public class MyTask extends AsyncTask<String, Void, List<Menu>> {  
  30. @Override 
  31. protected void onPreExecute() {  
  32. super.onPreExecute();  
  33. pd.show();  
  34. }  
  35. @Override 
  36. protected void onPostExecute(List<Menu> result) {  
  37. super.onPostExecute(result);  
  38. adapter.setData(result);  
  39. lv.setAdapter(adapter);  
  40. pd.dismiss();  
  41. }  
  42. @Override 
  43. protected List<Menu> doInBackground(String... params) {  
  44. String menuListStr = getListDishesInfo(params[0]);  
  45. return Utils.getInstance().parseMenusJSON(menuListStr);  
  46. }  
  47. }  
  48. private String getListDishesInfo(String sortId) {  
  49. // url  
  50. String url = HttpUtil.BASE_URL + "servlet/MenuInfoServlet?sortId=" 
  51. + sortId + "&flag=1";  
  52. // 查询返回结果  
  53. return HttpUtil.queryStringForPost(url);  
  54. }  
  55. }  

(2)、activity_main.xml的布局文件如下:

 

 
  1. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 
  2. xmlns:tools="http://schemas.android.com/tools" 
  3. android:layout_width="match_parent" 
  4. android:layout_height="match_parent" 
  5. android:background="#ffffff" 
  6. android:orientation="vertical" >  
  7. <ListView 
  8. android:id="@+id/lv" 
  9. android:layout_width="fill_parent" 
  10. android:layout_height="wrap_content" >  
  11. </ListView>  
  12. </LinearLayout>  

2、这是自定义的ListView的adapter的代码:

 

 
  1. package net.loonggg.adapter;  
  2. import java.util.List;  
  3. import net.loonggg.bean.Menu;  
  4. import net.loonggg.test.R;  
  5. import net.loonggg.util.ImageLoader;  
  6. import android.app.Activity;  
  7. import android.content.Context;  
  8. import android.view.LayoutInflater;  
  9. import android.view.View;  
  10. import android.view.ViewGroup;  
  11. import android.widget.BaseAdapter;  
  12. import android.widget.ImageView;  
  13. import android.widget.TextView;  
  14. public class MyAdapter extends BaseAdapter {  
  15. private List<Menu> list;  
  16. private Context context;  
  17. private Activity activity;  
  18. private ImageLoader imageLoader;  
  19. private ViewHolder viewHolder;  
  20. public MyAdapter(Context context) {  
  21. this.context = context;  
  22. this.activity = (Activity) context;  
  23. imageLoader = new ImageLoader(context);  
  24. }  
  25. public void setData(List<Menu> list) {  
  26. this.list = list;  
  27. }  
  28. @Override 
  29. public int getCount() {  
  30. return list.size();  
  31. }  
  32. @Override 
  33. public Object getItem(int position) {  
  34. return list.get(position);  
  35. }  
  36. @Override 
  37. public long getItemId(int position) {  
  38. return position;  
  39. }  
  40. @Override 
  41. public View getView(int position, View convertView, ViewGroup parent) {  
  42. if (convertView == null) {  
  43. convertView = LayoutInflater.from(context).inflate(  
  44. R.layout.listview_item, null);  
  45. viewHolder = new ViewHolder();  
  46. viewHolder.tv = (TextView) convertView.findViewById(R.id.item_tv);  
  47. viewHolder.iv = (ImageView) convertView.findViewById(R.id.item_iv);  
  48. convertView.setTag(viewHolder);  
  49. else {  
  50. viewHolder = (ViewHolder) convertView.getTag();  
  51. }  
  52. viewHolder.tv.setText(list.get(position).getDishes());  
  53. imageLoader.DisplayImage(list.get(position).getPicPath(), activity,  
  54. viewHolder.iv);  
  55. return convertView;  
  56. }  
  57. private class ViewHolder {  
  58. private ImageView iv;  
  59. private TextView tv;  
  60. }  
  61. }  

3、这是最重要的一部分代码,这就是异步加载图片的一个类,这里我就不解释了,代码中附有注释。代码如下:

 

 
  1. package net.loonggg.util;  
  2. import java.io.File;  
  3. import java.io.FileInputStream;  
  4. import java.io.FileNotFoundException;  
  5. import java.io.FileOutputStream;  
  6. import java.io.InputStream;  
  7. import java.io.OutputStream;  
  8. import java.net.HttpURLConnection;  
  9. import java.net.URL;  
  10. import java.util.Collections;  
  11. import java.util.Map;  
  12. import java.util.Stack;  
  13. import java.util.WeakHashMap;  
  14. import net.loonggg.test.R;  
  15. import android.app.Activity;  
  16. import android.content.Context;  
  17. import android.graphics.Bitmap;  
  18. import android.graphics.BitmapFactory;  
  19. import android.widget.ImageView;  
  20. /**  
  21. * 异步加载图片类  
  22.  
  23. * @author loonggg  
  24.  
  25. */ 
  26. public class ImageLoader {  
  27. // 手机中的缓存  
  28. private MemoryCache memoryCache = new MemoryCache();  
  29. // sd卡缓存  
  30. private FileCache fileCache;  
  31. private PicturesLoader pictureLoaderThread = new PicturesLoader();  
  32. private PicturesQueue picturesQueue = new PicturesQueue();  
  33. private Map<ImageView, String> imageViews = Collections  
  34. .synchronizedMap(new WeakHashMap<ImageView, String>());  
  35. public ImageLoader(Context context) {  
  36. // 设置线程的优先级  
  37. pictureLoaderThread.setPriority(Thread.NORM_PRIORITY - 1);  
  38. fileCache = new FileCache(context);  
  39. }  
  40. // 在找不到图片时,默认的图片  
  41. final int stub_id = R.drawable.stub;  
  42. public void DisplayImage(String url, Activity activity, ImageView imageView) {  
  43. imageViews.put(imageView, url);  
  44. Bitmap bitmap = memoryCache.get(url);  
  45. if (bitmap != null)  
  46. imageView.setImageBitmap(bitmap);  
  47. else {// 如果手机内存缓存中没有图片,则调用任务队列,并先设置默认图片  
  48. queuePhoto(url, activity, imageView);  
  49. imageView.setImageResource(stub_id);  
  50. }  
  51. }  
  52. private void queuePhoto(String url, Activity activity, ImageView imageView) {  
  53. // 这ImageView可能之前被用于其它图像。所以可能会有一些旧的任务队列。我们需要清理掉它们。  
  54. picturesQueue.Clean(imageView);  
  55. PictureToLoad p = new PictureToLoad(url, imageView);  
  56. synchronized (picturesQueue.picturesToLoad) {  
  57. picturesQueue.picturesToLoad.push(p);  
  58. picturesQueue.picturesToLoad.notifyAll();  
  59. }  
  60. // 如果这个线程还没有启动,则启动线程  
  61. if (pictureLoaderThread.getState() == Thread.State.NEW)  
  62. pictureLoaderThread.start();  
  63. }  
  64. /**  
  65. * 根据url获取相应的图片的Bitmap  
  66.  
  67. * @param url  
  68. * @return  
  69. */ 
  70. private Bitmap getBitmap(String url) {  
  71. File f = fileCache.getFile(url);  
  72. // 从SD卡缓存中获取  
  73. Bitmap b = decodeFile(f);  
  74. if (b != null)  
  75. return b;  
  76. // 否则从网络中获取  
  77. try {  
  78. Bitmap bitmap = null;  
  79. URL imageUrl = new URL(url);  
  80. HttpURLConnection conn = (HttpURLConnection) imageUrl  
  81. .openConnection();  
  82. conn.setConnectTimeout(30000);  
  83. conn.setReadTimeout(30000);  
  84. InputStream is = conn.getInputStream();  
  85. OutputStream os = new FileOutputStream(f);  
  86. // 将图片写到sd卡目录中去  
  87. ImageUtil.CopyStream(is, os);  
  88. os.close();  
  89. bitmap = decodeFile(f);  
  90. return bitmap;  
  91. catch (Exception ex) {  
  92. ex.printStackTrace();  
  93. return null;  
  94. }  
  95. }  
  96. // 解码图像和缩放以减少内存的消耗  
  97. private Bitmap decodeFile(File f) {  
  98. try {  
  99. // 解码图像尺寸  
  100. BitmapFactory.Options o = new BitmapFactory.Options();  
  101. o.inJustDecodeBounds = true;  
  102. BitmapFactory.decodeStream(new FileInputStream(f), null, o);  
  103. // 找到正确的缩放值。这应该是2的幂。  
  104. final int REQUIRED_SIZE = 70;  
  105. int width_tmp = o.outWidth, height_tmp = o.outHeight;  
  106. int scale = 1;  
  107. while (true) {  
  108. if (width_tmp / 2 < REQUIRED_SIZE  
  109. || height_tmp / 2 < REQUIRED_SIZE)  
  110. break;  
  111. width_tmp /= 2;  
  112. height_tmp /= 2;  
  113. scale *= 2;  
  114. }  
  115. // 设置恰当的inSampleSize可以使BitmapFactory分配更少的空间  
  116. // 用正确恰当的inSampleSize进行decode  
  117. BitmapFactory.Options o2 = new BitmapFactory.Options();  
  118. o2.inSampleSize = scale;  
  119. return BitmapFactory.decodeStream(new FileInputStream(f), null, o2);  
  120. catch (FileNotFoundException e) {  
  121. }  
  122. return null;  
  123. }  
  124. /**  
  125. * PictureToLoad类(包括图片的地址和ImageView对象)  
  126.  
  127. * @author loonggg  
  128.  
  129. */ 
  130. private class PictureToLoad {  
  131. public String url;  
  132. public ImageView imageView;  
  133. public PictureToLoad(String u, ImageView i) {  
  134. url = u;  
  135. imageView = i;  
  136. }  
  137. }  
  138. public void stopThread() {  
  139. pictureLoaderThread.interrupt();  
  140. }  
  141. // 存储下载的照片列表  
  142. class PicturesQueue {  
  143. private Stack<PictureToLoad> picturesToLoad = new Stack<PictureToLoad>();  
  144. // 删除这个ImageView的所有实例  
  145. public void Clean(ImageView image) {  
  146. for (int j = 0; j < picturesToLoad.size();) {  
  147. if (picturesToLoad.get(j).imageView == image)  
  148. picturesToLoad.remove(j);  
  149. else 
  150. ++j;  
  151. }  
  152. }  
  153. }  
  154. // 图片加载线程  
  155. class PicturesLoader extends Thread {  
  156. public void run() {  
  157. try {  
  158. while (true) {  
  159. // 线程等待直到有图片加载在队列中  
  160. if (picturesQueue.picturesToLoad.size() == 0)  
  161. synchronized (picturesQueue.picturesToLoad) {  
  162. picturesQueue.picturesToLoad.wait();  
  163. }  
  164. if (picturesQueue.picturesToLoad.size() != 0) {  
  165. PictureToLoad photoToLoad;  
  166. synchronized (picturesQueue.picturesToLoad) {  
  167. photoToLoad = picturesQueue.picturesToLoad.pop();  
  168. }  
  169. Bitmap bmp = getBitmap(photoToLoad.url);  
  170. // 写到手机内存中  
  171. memoryCache.put(photoToLoad.url, bmp);  
  172. String tag = imageViews.get(photoToLoad.imageView);  
  173. if (tag != null && tag.equals(photoToLoad.url)) {  
  174. BitmapDisplayer bd = new BitmapDisplayer(bmp,  
  175. photoToLoad.imageView);  
  176. Activity activity = (Activity) photoToLoad.imageView  
  177. .getContext();  
  178. activity.runOnUiThread(bd);  
  179. }  
  180. }  
  181. if (Thread.interrupted())  
  182. break;  
  183. }  
  184. catch (InterruptedException e) {  
  185. // 在这里允许线程退出  
  186. }  
  187. }  
  188. }  
  189. // 在UI线程中显示Bitmap图像  
  190. class BitmapDisplayer implements Runnable {  
  191. Bitmap bitmap;  
  192. ImageView imageView;  
  193. public BitmapDisplayer(Bitmap bitmap, ImageView imageView) {  
  194. this.bitmap = bitmap;  
  195. this.imageView = imageView;  
  196. }  
  197. public void run() {  
  198. if (bitmap != null)  
  199. imageView.setImageBitmap(bitmap);  
  200. else 
  201. imageView.setImageResource(stub_id);  
  202. }  
  203. }  
  204. public void clearCache() {  
  205. memoryCache.clear();  
  206. fileCache.clear();  
  207. }  
  208. }  

4、紧接着是几个实体类,一个是缓存到SD卡中的实体类,还有一个是缓存到手机内存中的实体类。代码如下:

(1)、缓存到sd卡的实体类:

 

 
  1. package net.loonggg.util;  
  2. import java.io.File;  
  3. import android.content.Context;  
  4. public class FileCache {  
  5. private File cacheDir;  
  6. public FileCache(Context context) {  
  7. // 找到保存缓存的图片目录  
  8. if (android.os.Environment.getExternalStorageState().equals(  
  9. android.os.Environment.MEDIA_MOUNTED))  
  10. cacheDir = new File(  
  11. android.os.Environment.getExternalStorageDirectory(),  
  12. "newnews");  
  13. else 
  14. cacheDir = context.getCacheDir();  
  15. if (!cacheDir.exists())  
  16. cacheDir.mkdirs();  
  17. }  
  18. public File getFile(String url) {  
  19. String filename = String.valueOf(url.hashCode());  
  20. File f = new File(cacheDir, filename);  
  21. return f;  
  22. }  
  23. public void clear() {  
  24. File[] files = cacheDir.listFiles();  
  25. for (File f : files)  
  26. f.delete();  
  27. }  
  28. }  

(2)、缓存到手机内存的实体类:

 

 
  1. package net.loonggg.util;  
  2. import java.lang.ref.SoftReference;  
  3. import java.util.HashMap;  
  4. import android.graphics.Bitmap;  
  5. public class MemoryCache {  
  6. private HashMap<String, SoftReference<Bitmap>> cache=new HashMap<String, SoftReference<Bitmap>>();  
  7. public Bitmap get(String id){  
  8. if(!cache.containsKey(id))  
  9. return null;  
  10. SoftReference<Bitmap> ref=cache.get(id);  
  11. return ref.get();  
  12. }  
  13. public void put(String id, Bitmap bitmap){  
  14. cache.put(id, new SoftReference<Bitmap>(bitmap));  
  15. }  
  16. public void clear() {  
  17. cache.clear();  
  18. }  
  19. }  

5、这个是输入输出流转换的类,及方法:

 

 
  1. package net.loonggg.util;  
  2. import java.io.InputStream;  
  3. import java.io.OutputStream;  
  4. public class ImageUtil {  
  5. public static void CopyStream(InputStream is, OutputStream os) {  
  6. final int buffer_size = 1024;  
  7. try {  
  8. byte[] bytes = new byte[buffer_size];  
  9. for (;;) {  
  10. int count = is.read(bytes, 0, buffer_size);  
  11. if (count == -1)  
  12. break;  
  13. os.write(bytes, 0, count);  
  14. }  
  15. catch (Exception ex) {  
  16. }  
  17. }  
  18. }  

到这里基本就完成了。

希望本文所述对大家Android程序设计有所帮助。

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表