最近在做android平板上的开发,其中涉及到高分辨率之下使用GridView的性能问题。在Android手机软件开发中,如果在ListView或者GridView上使用大数量Item,很多人都会想到ViewHolder......没错,ViewHolder非常适合用在ListView或者每行小于4个Item的GridView。但是如果是高分辨率的设备(android平板甚至android电视),每行包含4个以上Item的话,即使用了ViewHolder也依然卡。

如下图,每行9个Item,而且每个Item的图片都是从网络动态下载的,这时就比较考验GridView视图的优化了。

[置顶] Android平板、TV中对GridView的优化_第1张图片

本文提出的优化方法是:在getView()构建一个View列表(List<View>),把最近构建的View存起来,回退时直接从View列表中读取,而不是动态构建。使用这种方法有2个好处:

1.快速读取过去的Item;

2.直接保存View而不是Bitmap,避免了ImageView.setImageBitmaps()带来的延时。

当然坏处就是浪费内存,所以要设定一个上限,超过了就删掉最老的Item。
先来看看这种方法与ViewHolder的性能对比:


[置顶] Android平板、TV中对GridView的优化_第2张图片

100个Item往下滚到的三组数据对比,如上图:
“CacheAdapter 缓存50个Item”跟ViewHolderAdapter的速度很接近,由于CacheAdapter有缓存,所以会有1~2次快速读取Item(10~20个)的情况,而ViewHolder的每次读取Item速度比较平均。
“CacheAdapter 缓存75个Item”只在第一次往下滚动时消耗较长时间,第二次用了缓存的Item,所以速度快了很多。

[置顶] Android平板、TV中对GridView的优化_第3张图片

100个Item往上滚到的三组数据对比,如上图:

“CacheAdapter 缓存50个Item”比ViewHolderAdapter的速度略快,“CacheAdapter 缓存75个Item”依然是最快的。
总结:“CacheAdapter 缓存50个Item”速度与HolderView略快,读取最近的Item速度最快,缓存的Item越多速度越快。“CacheAdapter 缓存75个Item”占用内存最少,这是由于一部分图片下载失败,保存的Item的图片为空,实际上是缓存越多Item占用的内存越多。

PS:这里用到异步读取网络图片,成功下载的就占用较多内存,下载失败就占用较少内存,所以内存占用情况并不是一个时刻的绝对值,占用内存只用于参考.....

本文程序源码可以到http://www.rayfile.com/zh-cn/files/5ebf5666-958a-11e0-99ec-0015c55db73d/这里下载。

CacheAdapter.java是实现缓存Item的自定义Adapter,源码如下:

[java] view plain copy print ?
  1. /**
  2. * 使用列表缓存过去的Item
  3. * @author hellogv
  4. *
  5. */
  6. public class CacheAdapterextends BaseAdapter {
  7. public class Item {
  8. public String itemImageURL;
  9. public String itemTitle;
  10. public Item(String itemImageURL, String itemTitle) {
  11. this.itemImageURL = itemImageURL;
  12. this.itemTitle = itemTitle;
  13. }
  14. }
  15. private Context mContext;
  16. private ArrayList<Item> mItems =new ArrayList<Item>();
  17. LayoutInflater inflater;
  18. public CacheAdapter(Context c) {
  19. mContext = c;
  20. inflater = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
  21. }
  22. public void addItem(String itemImageURL, String itemTitle) {
  23. mItems.add(new Item(itemImageURL, itemTitle));
  24. }
  25. public int getCount() {
  26. return mItems.size();
  27. }
  28. public Item getItem(int position) {
  29. return mItems.get(position);
  30. }
  31. public long getItemId(int position) {
  32. return position;
  33. }
  34. List<Integer> lstPosition=new ArrayList<Integer>();
  35. List<View> lstView=new ArrayList<View>();
  36. List<Integer> lstTimes= new ArrayList<Integer>();
  37. long startTime=0;
  38. public View getView(int position, View convertView, ViewGroup parent) {
  39. startTime=System.nanoTime();
  40. if (lstPosition.contains(position) ==false) {
  41. if(lstPosition.size()>75)//这里设置缓存的Item数量
  42. {
  43. lstPosition.remove(0);//删除第一项
  44. lstView.remove(0);//删除第一项
  45. }
  46. convertView = inflater.inflate(R.layout.item, null);
  47. TextView text = (TextView) convertView.findViewById(R.id.itemText);
  48. ImageView icon = (ImageView) convertView.findViewById(R.id.itemImage);
  49. text.setText(mItems.get(position).itemTitle);
  50. new AsyncLoadImage().execute(new Object[] { icon,mItems.get(position).itemImageURL });
  51. lstPosition.add(position);//添加最新项
  52. lstView.add(convertView);//添加最新项
  53. } else
  54. {
  55. convertView = lstView.get(lstPosition.indexOf(position));
  56. }
  57. int endTime=(int) (System.nanoTime()-startTime);
  58. lstTimes.add(endTime);
  59. if(lstTimes.size()==10)
  60. {
  61. int total=0;
  62. for(int i=0;i<lstTimes.size();i++)
  63. total=total+lstTimes.get(i);
  64. Log.e("10个所花的时间:" +total/1000 +" μs",
  65. "所用内存:"+Runtime.getRuntime().totalMemory()/1024 +" KB");
  66. lstTimes.clear();
  67. }
  68. return convertView;
  69. }
  70. /**
  71. * 异步读取网络图片
  72. * @author hellogv
  73. */
  74. class AsyncLoadImage extends AsyncTask<Object, Object, Void> {
  75. @Override
  76. protected Void doInBackground(Object... params) {
  77. try {
  78. ImageView imageView=(ImageView) params[0];
  79. String url=(String) params[1];
  80. Bitmap bitmap = getBitmapByUrl(url);
  81. publishProgress(new Object[] {imageView, bitmap});
  82. } catch (MalformedURLException e) {
  83. Log.e("error",e.getMessage());
  84. e.printStackTrace();
  85. } catch (IOException e) {
  86. Log.e("error",e.getMessage());
  87. e.printStackTrace();
  88. }
  89. return null;
  90. }
  91. protected void onProgressUpdate(Object... progress) {
  92. ImageView imageView = (ImageView) progress[0];
  93. imageView.setImageBitmap((Bitmap) progress[1]);
  94. }
  95. }
  96. static public Bitmap getBitmapByUrl(String urlString)
  97. throws MalformedURLException, IOException {
  98. URL url = new URL(urlString);
  99. URLConnection connection = url.openConnection();
  100. connection.setConnectTimeout(25000);
  101. connection.setReadTimeout(90000);
  102. Bitmap bitmap = BitmapFactory.decodeStream(connection.getInputStream());
  103. return bitmap;
  104. }
  105. }
/** * 使用列表缓存过去的Item * @author hellogv * */ public class CacheAdapter extends BaseAdapter { public class Item { public String itemImageURL; public String itemTitle; public Item(String itemImageURL, String itemTitle) { this.itemImageURL = itemImageURL; this.itemTitle = itemTitle; } } private Context mContext; private ArrayList<Item> mItems = new ArrayList<Item>(); LayoutInflater inflater; public CacheAdapter(Context c) { mContext = c; inflater = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE); } public void addItem(String itemImageURL, String itemTitle) { mItems.add(new Item(itemImageURL, itemTitle)); } public int getCount() { return mItems.size(); } public Item getItem(int position) { return mItems.get(position); } public long getItemId(int position) { return position; } List<Integer> lstPosition=new ArrayList<Integer>(); List<View> lstView=new ArrayList<View>(); List<Integer> lstTimes= new ArrayList<Integer>(); long startTime=0; public View getView(int position, View convertView, ViewGroup parent) { startTime=System.nanoTime(); if (lstPosition.contains(position) == false) { if(lstPosition.size()>75)//这里设置缓存的Item数量 { lstPosition.remove(0);//删除第一项 lstView.remove(0);//删除第一项 } convertView = inflater.inflate(R.layout.item, null); TextView text = (TextView) convertView.findViewById(R.id.itemText); ImageView icon = (ImageView) convertView.findViewById(R.id.itemImage); text.setText(mItems.get(position).itemTitle); new AsyncLoadImage().execute(new Object[] { icon,mItems.get(position).itemImageURL }); lstPosition.add(position);//添加最新项 lstView.add(convertView);//添加最新项 } else { convertView = lstView.get(lstPosition.indexOf(position)); } int endTime=(int) (System.nanoTime()-startTime); lstTimes.add(endTime); if(lstTimes.size()==10) { int total=0; for(int i=0;i<lstTimes.size();i++) total=total+lstTimes.get(i); Log.e("10个所花的时间:" +total/1000 +" μs", "所用内存:"+Runtime.getRuntime().totalMemory()/1024 +" KB"); lstTimes.clear(); } return convertView; } /** * 异步读取网络图片 * @author hellogv */ class AsyncLoadImage extends AsyncTask<Object, Object, Void> { @Override protected Void doInBackground(Object... params) { try { ImageView imageView=(ImageView) params[0]; String url=(String) params[1]; Bitmap bitmap = getBitmapByUrl(url); publishProgress(new Object[] {imageView, bitmap}); } catch (MalformedURLException e) { Log.e("error",e.getMessage()); e.printStackTrace(); } catch (IOException e) { Log.e("error",e.getMessage()); e.printStackTrace(); } return null; } protected void onProgressUpdate(Object... progress) { ImageView imageView = (ImageView) progress[0]; imageView.setImageBitmap((Bitmap) progress[1]); } } static public Bitmap getBitmapByUrl(String urlString) throws MalformedURLException, IOException { URL url = new URL(urlString); URLConnection connection = url.openConnection(); connection.setConnectTimeout(25000); connection.setReadTimeout(90000); Bitmap bitmap = BitmapFactory.decodeStream(connection.getInputStream()); return bitmap; } }

其中if(lstPosition.size()>75)是设置缓存的Item数量的关键地方,这里缓存75个Item。

ViewHolderAdapter.java是实现ViewHolder加载Item的自定义Adapter,源码如下:

[java] view plain copy print ?
  1. /**
  2. * 使用ViewHolder加载Item
  3. * @author hellogv
  4. *
  5. */
  6. public class ViewHolderAdapterextends BaseAdapter {
  7. public class Item {
  8. public String itemImageURL;
  9. public String itemTitle;
  10. public Item(String itemImageURL, String itemTitle) {
  11. this.itemImageURL = itemImageURL;
  12. this.itemTitle = itemTitle;
  13. }
  14. }
  15. private Context mContext;
  16. private ArrayList<Item> mItems =new ArrayList<Item>();
  17. LayoutInflater inflater;
  18. public ViewHolderAdapter(Context c) {
  19. mContext = c;
  20. inflater = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
  21. }
  22. public void addItem(String itemImageURL, String itemTitle) {
  23. mItems.add(new Item(itemImageURL, itemTitle));
  24. }
  25. public int getCount() {
  26. return mItems.size();
  27. }
  28. public Item getItem(int position) {
  29. return mItems.get(position);
  30. }
  31. public long getItemId(int position) {
  32. return position;
  33. }
  34. static class ViewHolder {
  35. TextView text;
  36. ImageView icon;
  37. }
  38. List<Integer> lstTimes= new ArrayList<Integer>();
  39. long startTime=0;
  40. public View getView(int position, View convertView, ViewGroup parent) {
  41. startTime=System.nanoTime();
  42. ViewHolder holder;
  43. if (convertView == null) {
  44. convertView = inflater.inflate(R.layout.item, null);
  45. holder = new ViewHolder();
  46. holder.text = (TextView) convertView.findViewById(R.id.itemText);
  47. holder.icon = (ImageView) convertView.findViewById(R.id.itemImage);
  48. convertView.setTag(holder);
  49. } else {
  50. holder = (ViewHolder) convertView.getTag();
  51. }
  52. holder.text.setText(mItems.get(position).itemTitle);
  53. new AsyncLoadImage().execute(new Object[]{holder.icon,mItems.get(position).itemImageURL });
  54. int endTime=(int) (System.nanoTime()-startTime);
  55. lstTimes.add(endTime);
  56. if(lstTimes.size()==10)
  57. {
  58. int total=0;
  59. for(int i=0;i<lstTimes.size();i++)
  60. total=total+lstTimes.get(i);
  61. Log.e("10个所花的时间:" +total/1000 +" μs",
  62. "所用内存:"+Runtime.getRuntime().totalMemory()/1024 +" KB");
  63. lstTimes.clear();
  64. }
  65. return convertView;
  66. }
  67. /**
  68. * 异步读取网络图片
  69. * @author hellogv
  70. */
  71. class AsyncLoadImage extends AsyncTask<Object, Object, Void> {
  72. @Override
  73. protected Void doInBackground(Object... params) {
  74. try {
  75. ImageView imageView=(ImageView) params[0];
  76. String url=(String) params[1];
  77. Bitmap bitmap = CacheAdapter.getBitmapByUrl(url);
  78. publishProgress(new Object[] {imageView, bitmap});
  79. } catch (MalformedURLException e) {
  80. e.printStackTrace();
  81. } catch (IOException e) {
  82. e.printStackTrace();
  83. }
  84. return null;
  85. }
  86. protected void onProgressUpdate(Object... progress) {
  87. ImageView imageView = (ImageView) progress[0];
  88. imageView.setImageBitmap((Bitmap) progress[1]);
  89. }
  90. }
  91. }
/** * 使用ViewHolder加载Item * @author hellogv * */ public class ViewHolderAdapter extends BaseAdapter { public class Item { public String itemImageURL; public String itemTitle; public Item(String itemImageURL, String itemTitle) { this.itemImageURL = itemImageURL; this.itemTitle = itemTitle; } } private Context mContext; private ArrayList<Item> mItems = new ArrayList<Item>(); LayoutInflater inflater; public ViewHolderAdapter(Context c) { mContext = c; inflater = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE); } public void addItem(String itemImageURL, String itemTitle) { mItems.add(new Item(itemImageURL, itemTitle)); } public int getCount() { return mItems.size(); } public Item getItem(int position) { return mItems.get(position); } public long getItemId(int position) { return position; } static class ViewHolder { TextView text; ImageView icon; } List<Integer> lstTimes= new ArrayList<Integer>(); long startTime=0; public View getView(int position, View convertView, ViewGroup parent) { startTime=System.nanoTime(); ViewHolder holder; if (convertView == null) { convertView = inflater.inflate(R.layout.item, null); holder = new ViewHolder(); holder.text = (TextView) convertView.findViewById(R.id.itemText); holder.icon = (ImageView) convertView.findViewById(R.id.itemImage); convertView.setTag(holder); } else { holder = (ViewHolder) convertView.getTag(); } holder.text.setText(mItems.get(position).itemTitle); new AsyncLoadImage().execute(new Object[]{holder.icon,mItems.get(position).itemImageURL }); int endTime=(int) (System.nanoTime()-startTime); lstTimes.add(endTime); if(lstTimes.size()==10) { int total=0; for(int i=0;i<lstTimes.size();i++) total=total+lstTimes.get(i); Log.e("10个所花的时间:" +total/1000 +" μs", "所用内存:"+Runtime.getRuntime().totalMemory()/1024 +" KB"); lstTimes.clear(); } return convertView; } /** * 异步读取网络图片 * @author hellogv */ class AsyncLoadImage extends AsyncTask<Object, Object, Void> { @Override protected Void doInBackground(Object... params) { try { ImageView imageView=(ImageView) params[0]; String url=(String) params[1]; Bitmap bitmap = CacheAdapter.getBitmapByUrl(url); publishProgress(new Object[] {imageView, bitmap}); } catch (MalformedURLException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } return null; } protected void onProgressUpdate(Object... progress) { ImageView imageView = (ImageView) progress[0]; imageView.setImageBitmap((Bitmap) progress[1]); } } }

testPerformance.java是主程序,通过注释符就可以分别测试CacheAdapter与ViewHolderAdapter的性能,源码如下:

[java] view plain copy print ?
  1. public class testPerformanceextends Activity {
  2. /** Called when the activity is first created. */
  3. @Override
  4. public void onCreate(Bundle savedInstanceState) {
  5. super.onCreate(savedInstanceState);
  6. setContentView(R.layout.main);
  7. this.setTitle("android平板上的GridView视图缓存优化-----hellogv");
  8. GridView gridview = (GridView) findViewById(R.id.gridview);
  9. CacheAdapter adapter=new CacheAdapter(this);
  10. // ViewHolderAdapter adapter=new ViewHolderAdapter(this);
  11. gridview.setAdapter(adapter);
  12. String urlImage="";//请自己选择网络上的静态图片
  13. for(int i=0;i<100;i++)
  14. {
  15. adapter.addItem(urlImage, "第"+i+"项");
  16. }
  17. }
  18. }

本文转载自博客:http://blog.csdn.net/hellogv/article/details/6541286

更多相关文章

  1. Android Handler内存泄漏解决方法
  2. android 内存消耗分析
  3. 一定要注意 Android 内存泄漏问题
  4. 【Android】 保存图片到系统图库, 并立即显示在图库中
  5. Android 获得图片并解码成缩略图以减少内存消耗
  6. Android加载/处理超大图片神器!SubsamplingScaleImageView(subsam
  7. Android修改icon和roundIcon后,显示的图标还是默认的图片
  8. android中异步加载图片

随机推荐

  1. android 一行代码,快速实现图片验证码(附an
  2. android中Dialog居中显示
  3. android onSaveInstanceState的使用方法
  4. Android各版本代号/版本号/API级别
  5. Button简单实例1
  6. ubuntu16.04 x86_64下编译ffmpeg-4.0 for
  7. android强制隐藏输入法键盘(亲测可用,欢迎
  8. Android中点击按钮的事件处理实现步骤
  9. 反编译出错
  10. Android通过换载体实现再次辉煌