【Android开发经验】ViewHolder到底用什么修饰?static?final?static final?

来源:互联网 发布:js中获取json对象的值 编辑:程序博客网 时间:2024/06/06 19:43

转载请注明出处:http://blog.csdn.net/zhaokaiqiang1992

    现在大家都知道用ViewHolder来实现listview的优化了,但是,ViewHolder到底要用什么来修饰呢?这种修饰有什么意义呢?在一个ListView里面,存在多少个VIewHolder实例呢?为什么VIewHolder能够减少findview的次数,优化效率呢?为了弄清楚这个问题,我做了以下测试,测试代码很简单,就是下面这些

[java] view plaincopy在CODE上查看代码片派生到我的代码片
  1. public class MainActivity extends Activity {  
  2.   
  3.     public static int itemId = 1;  
  4.   
  5.     @Override  
  6.     protected void onCreate(Bundle savedInstanceState) {  
  7.         super.onCreate(savedInstanceState);  
  8.   
  9.         ListView listView = new ListView(this);  
  10.         listView.setAdapter(new MyAdapter());  
  11.         setContentView(listView);  
  12.   
  13.     }  
  14.   
  15.     private class MyAdapter extends BaseAdapter {  
  16.   
  17.         @Override  
  18.         public int getCount() {  
  19.             return 50;  
  20.         }  
  21.   
  22.         @Override  
  23.         public Object getItem(int position) {  
  24.             return position;  
  25.         }  
  26.   
  27.         @Override  
  28.         public long getItemId(int position) {  
  29.             return position;  
  30.         }  
  31.   
  32.         @Override  
  33.         public View getView(int position, View convertView, ViewGroup parent) {  
  34.   
  35.             ViewHolder viewHolder = null;  
  36.   
  37.             if (convertView == null) {  
  38.                 convertView = getLayoutInflater().inflate(R.layout.item,  
  39.                         parent, false);  
  40.                 viewHolder = new ViewHolder();  
  41.                 viewHolder.tv = (TextView) convertView.findViewById(R.id.tv);  
  42.                 convertView.setTag(viewHolder);  
  43.             } else {  
  44.                 viewHolder = (ViewHolder) convertView.getTag();  
  45.             }  
  46.   
  47.             viewHolder.tv.setText("item" + position);  
  48.   
  49.             Log.d("TAG""position=" + position + "---" + viewHolder.toString());  
  50.   
  51.             return convertView;  
  52.         }  
  53.   
  54.     }  
  55.   
  56.       
  57.     private  class ViewHolder {  
  58. //  private static class ViewHolder {  
  59. //  private final class ViewHolder {  
  60. //  private static final class ViewHolder {  
  61.   
  62.         private int id;  
  63.   
  64.         public ViewHolder() {  
  65.             id = itemId;  
  66.             Log.d("TAG""ViewHolder" + id);  
  67.             itemId++;  
  68.         }  
  69.   
  70.         TextView tv;  
  71.   
  72.         @Override  
  73.         public String toString() {  
  74.             return "--------id=" + id;  
  75.         }  
  76.   
  77.     }  
  78.   
  79. }  

    我简单说一下代码是什么意思,ViewHolder有一个成员变量id,在toString()使用,用来区分不同的ViewHolder,在构造函数中,对id进行赋值,itemId是一个静态变量,每初始化一次就+1,我们可以根据构造函数的打印次数,来计算ViewHolder的实例化次数,根据toString()可以来判断到底是使用了哪一个ViewHolder。getVIew中的写法是固定的,下面是测试结果:


    private class ViewHolder修饰,界面内可见的item数量是9个,ViewHolder初始化了10次,之后开始复用ViewHolder。


    private static class ViewHolder修饰,界面内可见的item数量是9个,ViewHolder初始化了10次,之后开始复用ViewHolder。


    private final class ViewHolder修饰,界面内可见的item数量是9个,ViewHolder初始化了10次,之后开始复用ViewHolder。


    private static final class ViewHolder修饰,完全一样!



    其实一上来我就贴一张图就ok了,因为这几种完全一样,不管用什么修饰,ViewHolder都会初始化当前界面可见item的数量+1次,和convertView的实例化次数是一样的,因此,我么不管使用什么修饰都是没问题的,也是没必要的。

    static修饰类,在这里是静态内部类,并不是说只存在一个实例,而是可以访问外部类的静态变量,final修饰类则是不让该类继承,我们这里使用final毫无根据,所以,以后写ViewHolder的时候,可以不纠结了,加什么加啊,什么都不用加!



----------------------------------------------------------------------------------------

今天又请教了公司一个大神,他的回答如下:

这篇文章中没有提到加载时机的问题吧。在我的理解中:静态内部类主要作用就是,内部类是否需要隔离“外部类的this对象(指针)”。内部类是有 this 指针的,可以“直接”访问外部类的 成员变量和成员函数(包括私有的成员)。而静态内部类,没有这个this指针,所以无法“直接”调用。

个人的习惯,我通常会把内部静态类,作为一个单独的java文件。

但是Android代码中,却经常使用的Builder,LayoutParams都是以静态内部类的方式存在啊。我们可以根据实际使用情况来效仿。

另外还要提一点,内部静态类(特别是私有内部静态类),在代码混淆上效果更好。会增加反编译的难度。

-------------------------------------------------------
对于这篇文章,我理解。ViewHolder的构造和复用,与静不静态没关。核心是ListView(AdapterView),通过getView(int position, View convertView, ViewGroup parent) 的convertView会为开发者传入一个可以复用的对象。开发者需要利用该对象,减少应用内存的消耗。


如果从减少内存消耗的角度来开。我认为ViewHolder还是应该修饰成static比较好。这样ViewHolder中可以减少MainActivity的this指针,由于减少了一个this指针的引用,也会对MainActivity的引用计数大大减少。MainActivity的this指针继承于android的Context上下文,对于Context的回收遗漏,是Android内存管理中很大的问题。我们减少了对Context的引用,可以更容易减少Context引用计数出现问题。


0 0