listview
Aviewthatshowsitemsinaverticallyscrollinglist。
一個顯示一個垂直的滾動子項目的列表視圖在android開發中,使用listview的地方很多,用它來展現數據,成一個垂直的視圖。使用listview是標準的適配器模式,用資料--,介面--xml以及適配器--adapter,資料被適配器按照需要的方式展現出來,xml描寫了資料如何展現,activity中控制這些活動。
其中使用自訂的adapter,會要重寫getView方法,在getView方法產生給使用者item的視圖以及資料。
請參考圖:
這裡有一個最佳化的地方,就是重複使用view,這樣減少記憶體消耗,同時加快item載入速度。
在getView中優化的地方,大家想必都非常狀況,以下我總結了三種最佳化的寫法,請大家指正。
第一:
重複使用了convertView,很大程度的減少了記憶體的消耗。透過判斷convertView是否為null,是的話就需要產生一個視圖出來,然後給這個視圖數據,最後將這個視圖返回給底層,呈獻給用戶。
特點:如果目前的convertView為null,則透過LayoutInflat產生一個view。
ViewCode publicViewgetView(intposition,ViewconvertView,ViewGroupparent) { if(convertView==null) { convertView=LayoutInflater.from(context).inflate(R.layout.section_list_item1,null); } TextViewtv_name=(TextView)convertView.findViewById(R.id.contact_contactinfoitem_tv_name); TextViewtv_phone=(TextView)convertView.findViewById(R.id.contact_contactinfoitem_tv_phoneNum); ContactInfo1confo=contacts.get(position); if(confo!=null){//toseteveryitem'stext tv_name.setText(confo.getContactName()); tv_phone.setText(confo.getContact_Phone()); } returnconvertView; }
第二:
上面的寫法會有一個缺點,就是每次在getVIew的時候,都需要重新的findViewById,重新找到控制項,然後進行控制項的賦值以及事件對應設定。這樣其實在做重複的事情,因為的geiview中,其實包含有這些控件,而且這些控件的id還都是一樣的,也就是其實只要在view中findViewById一次,後面無需要每次都要findViewById了。
下面給出第二種寫法
寫發的特點,通常有一個內部類classViewHolder,這個ViewHolder,用來標識view中一些控件,方便進行一些事件相應操作的設置,比如onClick等等,這樣可以不用每次都要findViewById了,減少了效能的消耗。同時重用了convertView,很大程度上的減少了記憶體的消耗。
ViewCode publicViewgetView(intposition,ViewconvertView,ViewGroupparent) { ViewHolderholder; if(convertView==null){ convertView=LayoutInflater.from(context).inflate(R.layout.section_list_item1,null); holder=newViewHolder(); holder.tv_name=(TextView)convertView.findViewById(R.id.contact_contactinfoitem_tv_name); holder.tv_phone=(TextView)convertView.findViewById(R.id.contact_contactinfoitem_tv_phoneNum); convertView.setTag(holder); } else { holder=(ViewHolder)convertView.getTag(); } ContactInfo1confo=contacts.get(position); Log.i("my","confo"+confo.getContactName()); if(confo!=null){//toseteveryitem'stext holder.tv_name.setText(confo.getContactName()); holder.tv_phone.setText(confo.getContact_Phone()); } returnconvertView; } classViewHolder { TextViewtv_name,tv_phone; }
第三:
個人覺得這個寫法是最舒服的,最舒服的意思是看著程式碼有一種很爽,看的很清晰。
特點,使用了內部類別classViewHolder、重複使用了convertView。
區別第二種寫法是,使用了一個臨時變數Viewview=convertView,然後修改view,最後返回view
ViewCode @Override publicViewgetView(intposition,ViewconvertView,ViewGroupparent) { Viewview=convertView; ViewHolderholder; if(view==null){ view=LayoutInflater.from(context).inflate(R.layout.section_list_item1,null); holder=newViewHolder(); holder.tv_name=(TextView)view.findViewById(R.id.contact_contactinfoitem_tv_name); holder.tv_phone=(TextView)view.findViewById(R.id.contact_contactinfoitem_tv_phoneNum); view.setTag(holder); } else { holder=(ViewHolder)view.getTag(); } ContactInfo1confo=contacts.get(position); Log.i("my","confo"+confo.getContactName()); if(confo!=null){//toseteveryitem'stext holder.tv_name.setText(confo.getContactName()); holder.tv_phone.setText(confo.getContact_Phone()); } returnview; } classViewHolder { TextViewtv_name,tv_phone; }
以上就是集中寫法,供新手學習和總結。
原始碼如下:LisViewTest.zip
根據樓下朋友提供的建議,發現還有最佳化的地方,最新更新如下:
ViewCode @Override publicViewgetView(intposition,ViewconvertView,ViewGroupparent) { Viewview=convertView; ViewHolderholder; if(view==null){ view=LayoutInflater.from(context).inflate(R.layout.section_list_item1,null); holder=newViewHolder(); holder.tv_name=(TextView)view.findViewById(R.id.contact_contactinfoitem_tv_name); holder.tv_phone=(TextView)view.findViewById(R.id.contact_contactinfoitem_tv_phoneNum); view.setTag(holder); } else { holder=(ViewHolder)view.getTag(); } ContactInfo1confo=contacts.get(position); Log.i("my","confo"+confo.getContactName()); if(confo!=null){//toseteveryitem'stext holder.tv_name.setText(confo.getContactName()); holder.tv_phone.setText(confo.getContact_Phone()); } returnview; } <fontcolor="\"#0000ff\""></font>staticclassViewHolder { TextViewtv_name,tv_phone; }
注意:staticclassViewHolder
這裡設定ViewHolder為static,也就是靜態的,靜態類別只會靜態在第一次加載時會耗費比較長時間,但是後面就可以很好幫助加載,同時保證了內存中只有一個ViewHolder,節省了內存的開銷。