convertView,这个是Android在为我们而做的缓存机制。

ListView与Adapter使用要点

博客分类:  Android

 View getView(int position, View convertView, ViewGroup parent);

  /**

     * Get a View that displays the data at the specified position in the data set. You can either

     * create a View manually or inflate it from an XML layout file. When the View is inflated, the

     * parent View (GridView, ListView...) will apply default layout parameters unless you use

     * {@link android.view.LayoutInflater#inflate(int, android.view.ViewGroup, boolean)}

     * to specify a root view and to prevent attachment to the root.

     *

     * @param position The position of the item within the adapter's data set of the item whose view

     *        we want.

     * @param convertView The old view to reuse复用, if possible. Note: You should check that this view

     *        is non-null and of an appropriate type before using. If it is not possible to convert

     *        this view to display the correct data, this method can create a new view.

     *        Heterogeneous lists can specify their number of view types, so that this View is

     *        always of the right type (see {@link #getViewTypeCount()} and

     *        {@link #getItemViewType(int)}).

     * @paramparent  :  生成的view最终依附 The parent that this view will eventually be attached to

     * @return a View   corresponding to the data at the specified position.

     */ 

 

项目用到ListView,由于要用到 ImageView ,图片源不是在资源里面的,没法使用资源 ID,因此无法直接使用SimpleAdapter,要自己写一个Adapter。 在使用ListView和Adapter需要注意以下几点:

 

1. Adapter.getView()

 

public View getView(int position, View convertView , ViewGroup parent){...}

 

这个方法就是用来获得指定位置要显示的View。官网解释如下:

Get a View that displays the data at the specified position in the data set. You can either create a View manually or inflate it from an XML layout file.

 

当要显示一个View就调用一次这个方法。这个方法是ListView性能好坏的关键。方法中有个convertView,这个是Android在为我们而做的缓存机制。

ListView中每个item都是通过getView返回并显示的,假如item有很多个,那么重复创建这么多对象来显示显然是不合理。因此,Android提供了Recycler,将没有正在显示的item放进RecycleBin,然后在显示新视图时从RecycleBin中复用这个View。

 

Recycler的工作原理大致如下:

假设屏幕最多能看到11个item,那么当第1个item滚出屏幕,这个item的View进入RecycleBin中,第12个要出现前,通过getView从回收站(RecycleBin)中重用这个View,然后设置数据,而不必重新创建一个View。

 

我们用Android提供的APIDemos来验证这个过程:

 

先看关键代码:

Java代码  收藏代码
  1. public View getView(int position, View convertView, ViewGroup parent) {  
  2.     // A ViewHolder keeps references to children views to avoid unneccessary calls  
  3.     // to findViewById() on each row.  
  4.     ViewHolder holder;  
  5.     // When convertView is not null, we can reuse it directly, there is no need  
  6.     // to reinflate it. We only inflate a new View when the convertView supplied  
  7.     // by ListView is null.  
  8.     if (convertView == null) {  
  9.         convertView = mInflater.inflate(R.layout.list_item_icon_text, null);  
  10.         Log.v("tag""positon " + position + " convertView is null, " + "new: " + convertView);  
  11.         // Creates a ViewHolder and store references to the two children views  
  12.         // we want to bind data to.  
  13.         holder = new ViewHolder();  
  14.         holder.text = (TextView) convertView.findViewById(R.id.text);  
  15.         holder.icon = (ImageView) convertView.findViewById(R.id.icon);  
  16.         convertView.setTag(holder);  
  17.     } else {  
  18.         // Get the ViewHolder back to get fast access to the TextView  
  19.         // and the ImageView.  
  20.         holder = (ViewHolder) convertView.getTag();  
  21.         Log.v("tag""positon " + position + " convertView is not null, "  + convertView);  
  22.     }  
  23.     // Bind the data efficiently with the holder.  
  24.     holder.text.setText(DATA[position]);  
  25.     holder.icon.setImageBitmap((position & 1) == 1 ? mIcon1 : mIcon2);  
  26.     return convertView;  
  27. }  
  28.   
  29.  static class ViewHolder {  
  30.     TextView text;  
  31.     ImageView icon;  
  32. }  

 

效果图:

 

可以看到,一打开Activity,看到10个item.

我们看看Log信息:

 

可以看出,每次convertView都是null, 都是新建一个View来显示的。

当我们向下滑动,如下图,

  

 

由于item0和item10都显示一半,所以item10也是新建出来,但是当要显示item11的时候,由于item0已经不在屏幕上,所以item11复用了item0的实例。可以从以下Log信息看出:

我们分析Log信息,可以看出item11的对象是item0, item12的对象是item1,如此类推。

这样,通过复用convertView,就可以避免每次都新建View,节省内存而且优化ListView的滑动效果。

 

 

2.  ListView的Layout XML

 

除了上述说的,还有一个要点就是ListView在Layout XML中的描述。

先看问题:

有时,我们可能会看到一打开ListView,getView会重复调用好次(假设屏幕最多可以看到6个item ),如下图:

一直重复 0-6,  0-5,0-5, 0-5,0-5, 0-5。而且,convertView一开始都是同一个View,这个是因为ListView的

android:layout_height=" wrap_content"。

我们修改为android:layout_height="fill_parent" , Log信息如下:

可以看出,修改之后ListView的getView调用恢复和Recycler的行为一致。

至于为什么使用wrap_content会出现重复调用的情况,我还没有研究过。不过初步觉得是因为在Android描绘ListView的时候,由于不清楚高度,所以使用一个item去试探ListView在屏幕中的最大高度所引起。希望有知道的朋友能够告诉,先谢谢了!

 

最后,如果上面有什么地方说错的话,希望能够指出,互相进步嘛。

 

补充:

在接着使用ListView的时候,又发现一个很奇怪的现象。调用notifyDataSetChanged()之后,ListView在重新getView()时,所有的convertView的顺序都逆序了。请看下面截图:

 

 

这应该是由于recycleBin是stack结构而引起。

 

 

其它:

1.  Disable divider:
android:divider="#00000000"
android:dividerHeight="0dp"
            
2. Disable ListView selector:
convertView.setOnClickListener(null);

如果只是要去掉颜色,可以用android:listSelector="#00000000"
            
3. Disable header divider:
android:headerDividersEnabled="false"

 

4. getItemViewType(int)与getItemViewType(int)

getItemViewType(int) can not return int value larger than getViewTypeCount().
Otherwise you will get java.lang.ArrayIndexOutOfBoundsException at android.widget.AbsListView$RecycleBin.addScrapView(AbsListView.java:3523)

 

ListView会根据不同的ViewType返回相应type的convertView.

一般写法:

getView() {

switch (getItemViewType(position)) {

    case type1:

        if(convertView == null) {

        } else {

        }

        break;

    case type2:

    default:

        if(convertView == null) {

        } else {

        }

        break;

}

return convertView;

}

 

getItemViewType(int position) {

    // 根据场景,一般有:

    // 1. 不同的item type对应的position是固定的,那么ListView的data可以分别存放

    // 2. 不同的item type对应的position是不固定的,那么可以把ListView的data统一放在List<Object>中,

    //  然后使用instanceof来判断Object的类型进而区分position对应的view type.

}

 

展开阅读全文

这个是做什么用的?

02-26

function SetPassword()rnrnrn pas = window.document.getalbpass.p.value;rn var albumid = window.document.getalbpass.albumid.valuern var uin = window.document.getalbpass.uin.valuern var showtype = window.document.getalbpass.showtype.valuern if(pas=="")rn rn alert("请输入正确的密码!");rn return;rn rnrn var mdpas = MD5(pas);rn /*rn var str = document.URL;rn var i = str.lastIndexOf("?url=");rn if(i!==-1)rn rn str = str.substring(i+5, str.length);rn str = str + "&albpas=" + mdpas;rn window.opener.location.href=str;rn window.close();rn rn elsern window.opener.refresh();rn */rn// if( showtype == 2 )rn window.location.href = "/cgi-bin/qzone/cgi_photo_bigview?albumid="+albumid+"&uin="+uin+"&albpas=" + mdpas;rn// elsern// window.location.href = "/cgi-bin/qzone/cgi_browse_album?albumid="+albumid+"&uin="+uin+"&albpas=" + mdpas;rn rn //window.close();rn return ;rnrnrn/* MD5 Message-Digest Algorithm - JavaScriptrn' MODIFICATION HISTORY:rn' 1.0 16-Feb-2001 - Phil Fresle (sales@frez.co.uk) - Initial Version (VB/ASP code)rn' 1.0 21-Feb-2001 - Enrico Mosanghini (erik504@yahoo.com) - JavaScript portingrn*/rnfunction MD5(sMessage) rnrn function RotateLeft(lValue, iShiftBits) rn rn return (lValue<>>(32-iShiftBits)); rn rnrn function AddUnsigned(lX,lY) rn rn var lX4,lY4,lX8,lY8,lResult;rn lX8 = (lX & 0x80000000);rn lY8 = (lY & 0x80000000);rn lX4 = (lX & 0x40000000);rn lY4 = (lY & 0x40000000);rn lResult = (lX & 0x3FFFFFFF)+(lY & 0x3FFFFFFF);rn if (lX4 & lY4) return (lResult ^ 0x80000000 ^ lX8 ^ lY8);rn if (lX4 | lY4) rn if (lResult & 0x40000000) return (lResult ^ 0xC0000000 ^ lX8 ^ lY8);rn else return (lResult ^ 0x40000000 ^ lX8 ^ lY8);rn else return (lResult ^ lX8 ^ lY8);rn rn rn function F(x,y,z) return (x & y) | (( ~x) & z); rnrn function G(x,y,z) return (x & z) | (y & (~z)); rnrn function H(x,y,z) return (x ^ y ^ z); rn rn function I(x,y,z) return (y ^ (x | (~z))); rn rn function FF(a,b,c,d,x,s,ac) rn a = AddUnsigned(a, AddUnsigned(AddUnsigned(F(b, c, d), x), ac));rn return AddUnsigned(RotateLeft(a, s), b);rn rnrn function GG(a,b,c,d,x,s,ac) rn a = AddUnsigned(a, AddUnsigned(AddUnsigned(G(b, c, d), x), ac));rn return AddUnsigned(RotateLeft(a, s), b);rn rnrn function HH(a,b,c,d,x,s,ac) rn a = AddUnsigned(a, AddUnsigned(AddUnsigned(H(b, c, d), x), ac));rn return AddUnsigned(RotateLeft(a, s), b);rn rnrn function II(a,b,c,d,x,s,ac) rn a = AddUnsigned(a, AddUnsigned(AddUnsigned(I(b, c, d), x), ac));rn return AddUnsigned(RotateLeft(a, s), b);rn rnrn function ConvertToWordArray(sMessage) rn var lWordCount;rn var lMessageLength = sMessage.length;rn var lNumberOfWords_temp1=lMessageLength + 8;rn var lNumberOfWords_temp2=(lNumberOfWords_temp1-(lNumberOfWords_temp1 % 64))/64;rn var lNumberOfWords = (lNumberOfWords_temp2+1)*16;rn var lWordArray=Array(lNumberOfWords-1);rn var lBytePosition = 0;rn var lByteCount = 0;rn while ( lByteCount < lMessageLength ) rn lWordCount = (lByteCount-(lByteCount % 4))/4;rn lBytePosition = (lByteCount % 4)*8;rn lWordArray[lWordCount] = (lWordArray[lWordCount] | (sMessage.charCodeAt(lByteCount)<>>29;rn return lWordArray;rn rnrn function WordToHex(lValue) rn var WordToHexValue="",WordToHexValue_temp="",lByte,lCount;rn for (lCount = 0;lCount<=3;lCount++) rn lByte = (lValue>>>(lCount*8)) & 255;rn WordToHexValue_temp = "0" + lByte.toString(16);rn WordToHexValue = WordToHexValue + WordToHexValue_temp.substr(WordToHexValue_temp.length-2,2);rn rn return WordToHexValue;rn rnrn var x=Array();rn var k,AA,BB,CC,DD,a,b,c,drn var S11=7, S12=12, S13=17, S14=22;rn var S21=5, S22=9 , S23=14, S24=20;rn var S31=4, S32=11, S33=16, S34=23;rn var S41=6, S42=10, S43=15, S44=21;rn // Steps 1 and 2. Append padding bits and length and convert to wordsrn x = ConvertToWordArray(sMessage);rn // Step 3. Initialisern a = 0x67452301; b = 0xEFCDAB89; c = 0x98BADCFE; d = 0x10325476;rn // Step 4. Process the message in 16-word blocksrn for (k=0;k 论坛

没有更多推荐了,返回首页