利用convertView优化ListView和GirdView的性能 (Adapter的使用)

这里提到的ListView只是作为一个典型代表 其实在Android中 采用类似Adapter机制的GridView等都是可以适用的 而ListView应该是用得最多的 所以就以它来举例

大家都知道 将ListView和Adapter绑定以后 其实也就是将数据源和控件显示绑定在一起 而每次需要显示ListView的时候 里面的item其实是Adapter提供的 通过的就是重要的getView()方法 而做法也是在这上面进行

先来看一下基本的getView写法

Java代码 收藏代码
  1. public View getView(int position, View convertView, ViewGroup parent) {
  2. View view = new View();
  3. //通过inflate等找到布局 然后findViewById等 设置各个显示的item
  4. return view;
  5. }

而在ListView滑动的过程中 很容易就会发现每次getView被执行 都会new出一个View对象 长此以往会产生很大的消耗 特别当item中还有Bitmap等 甚至会造成OOM的错误导致程序崩溃

在看getView提供的参数时 可能已经注意到了 有一个参数View convertView 而这个convertView其实就是最关键的部分了 原理上讲 当ListView滑动的过程中 会有item被滑出屏幕 而不再被使用 这时候Android会回收这个条目的view 这个view也就是这里的convertView

在上面的做法中 当item1被移除屏幕的时候 我们会重新new一个View给新显示的item_new 而如果使用了这个convertView 我们其实可以复用它 这样就省去了new View的大量开销

下面就是使用convertView后的情况

Java代码 收藏代码
  1. public View getView(int position, View convertView, ViewGroup parent) {
  2. View view = null;
  3. if (convertView != null) {
  4. view = convertView;
  5. //复用了回收的view 只需要直接作内容填充的修改就好了
  6. } else {
  7. view = new Xxx(...);
  8. //没有供复用的view 按一般的做法新建view
  9. }
  10. return view;
  11. }

这样一来 就避免了反复创建大量view的问题了

但是上面的仍然有缺陷 当我们的ListView中填充的item有多种形式时 比如微博中 有的item中包含图片 有的item包含视频 那么必然的 我们需要用到2种item的布局方式

此时如果只是单纯判断convert是否存在 会造成回收的view不符合你当前需要的布局 而类似转换失败出错退出

这里要提到Adapter中的另外2个方法:

public int getItemViewType(int position) {}

public int getViewTypeCount() {}

从方法名上 就可以比较明显的明白这2个的作用

下面附上一个demo代码

Java代码 收藏代码
  1. class MyAdapter extends BaseAdapter{
  2. Context mContext;
  3. LinearLayout linearLayout = null;
  4. LayoutInflater inflater;
  5. TextView tex;
  6. final int VIEW_TYPE = 2;
  7. final int TYPE_1 = 0;
  8. final int TYPE_2 = 1;
  9. public MyAdapter(Context context) {
  10. mContext = context;
  11. inflater = LayoutInflater.from(mContext);
  12. }
  13. @Override
  14. public int getCount() {
  15. return listString.size();
  16. }
  17. //每个convert view都会调用此方法,获得当前所需要的view样式
  18. @Override
  19. public int getItemViewType(int position) {
  20. int p = position%6;
  21. if(p == 0)
  22. return TYPE_1;
  23. else if(p < 3)
  24. return TYPE_2;
  25. else
  26. return TYPE_1;
  27. }
  28. @Override
  29. public int getViewTypeCount() {
  30. return 2;
  31. }
  32. @Override
  33. public Object getItem(int arg0) {
  34. return listString.get(arg0);
  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. viewHolder1 holder1 = null;
  43. viewHolder2 holder2 = null;
  44. int type = getItemViewType(position);
  45. //无convertView,需要new出各个控件
  46. if(convertView == null)
  47. {
  48. //按当前所需的样式,确定new的布局
  49. switch(type)
  50. {
  51. case TYPE_1:
  52. convertView = inflater.inflate(R.layout.listitem1, parent, false);
  53. holder1 = new viewHolder1();
  54. holder1.textView = (TextView)convertView.findViewById(R.id.textview1);
  55. holder1.checkBox = (CheckBox)convertView.findViewById(R.id.checkbox);
  56. convertView.setTag(holder1);
  57. break;
  58. case TYPE_2:
  59. convertView = inflater.inflate(R.layout.listitem2, parent, false);
  60. holder2 = new viewHolder2();
  61. holder2.textView = (TextView)convertView.findViewById(R.id.textview2);
  62. holder2.imageView = (ImageView)convertView.findViewById(R.id.imageview);
  63. convertView.setTag(holder2);
  64. break;
  65. }
  66. }
  67. else
  68. {
  69. //有convertView,按样式,取得不用的布局
  70. switch(type)
  71. {
  72. case TYPE_1:
  73. holder1 = (viewHolder1) convertView.getTag();
  74. break;
  75. case TYPE_2:
  76. holder2 = (viewHolder2) convertView.getTag();
  77. break;
  78. }
  79. //设置资源
  80. switch(type)
  81. {
  82. case TYPE_1:
  83. holder1.textView.setText(Integer.toString(position));
  84. holder1.checkBox.setChecked(true);
  85. break;
  86. case TYPE_2:
  87. holder2.textView.setText(Integer.toString(position));
  88. holder2.imageView.setBackgroundResource(R.drawable.icon);
  89. break;
  90. }
  91. }
  92. return convertView;
  93. }
  94. }
  95. //各个布局的控件资源
  96. class viewHolder1{
  97. CheckBox checkBox;
  98. TextView textView;
  99. }
  100. class viewHolder2{
  101. ImageView imageView;
  102. TextView textView;
  103. }

这里对于每个View使用了一个viewHolder来控制其内部的子item

还有一个需要注意的地方是使用了setTag和getTag的方法 将holder绑定到了view上 也算一种技巧

以上基本就是主要的内容了 下面再补充实际操作当中的一些Tips

*如果convertView上用Type区分有些繁琐 或者不需要那么复杂 只是很少有出现不同的情况 那么还可以在取得convertView后 通过java提供的 instanceof 来判断是否可以强转 如果不能强转 就去新建一个View的做法 但是其实这种做法并不规范 所以还是推荐上面的做法

*第二个是关于ListView 对于纯色的item背景 其实可以直接设置BackgroundColor 而不要使用图片 这一部分其实可以有不小的提升 同样的 对于任何纯色的背景 应该尽量去设置RGB颜色 而不是全用一张图片做背景

转自:http://johncookie.iteye.com/blog/1250049

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值