編輯:關於Android編程
Android中萬能的適配器的詳細講解
在Android開發中,適配器的用處是非常大的,尤其是效率優化方面。除了使用ViewHolder復用View之外,如果存在很多的ListView或者是一個ListView中存在很多的View組件,那對代碼的閱讀不是很好的。考慮到優化以及共通方面,我封裝了ViewHolder類以及將Adapter類封裝成共通的了,將對以後的開發帶來很大的方便。
適配器中提高性能優化如下:
1.利用convertView
利用Android的Recycler機制,利用convertView來重新回收View,效率有了本質提高。View的每次創建是比較耗時的,因此對於getview方法傳入的convertView應充分利用 != null的判斷 。
2.使用ViewHolder
ViewHolder將需要緩存的view封裝好,convertView的setTag才是將這些緩存起來供下次調用。 當你的listview裡布局多樣化的時候 viewholder的作用體現明顯,效率再一次提高。 View的findViewById()方法也是比較耗時的,因此需要考慮只調用一次,之後就用View.getTag()方法來獲得ViewHolder對象。
3.優雅的使用ViewHolder
使用ViewHolder時,每次一遍一遍的findViewById,一遍一遍在ViewHolder裡面添加View的定義,view一多,是不是感覺煩爆了,base-adapter-helper這個類庫似乎完美的解決了這個問題。
其設計思想是使用 SparseArray來存儲view的引用,代替了原本的ViewHolder,不用聲明一大堆View,簡潔明了。
(1).ViewHolder類的封裝如下:ViewHolder類:
package com.chengdong.su.baseadapter.util;
import android.content.Context;
import android.util.Log;
import android.util.SparseArray;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.TextView;
/**
* the common object of the ViewHolder
*
* @author scd
*
*/
public class ViewHolder {
/** the object of the TAG */
private String TAG = getClass().getSimpleName();
/** the object of the view */
private SparseArray mViews;
/** the object of the position */
private int mPosition;
/** the object of the converview */
private View mConvertView;
/***
* 構造方法
*
* @param context
* @param parent
* @param layoutId
* @param position
*/
public ViewHolder(Context context, ViewGroup parent, int layoutId,int position) {
super();
this.mPosition = position;
this.mViews = new SparseArray();
mConvertView = LayoutInflater.from(context).inflate(layoutId, parent,
false);
mConvertView.setTag(this);
}
/**
* get the object of the ViewHolder
*
* @param context
* @param convertView
* @param parent
* @param layoutId
* @param position
* @return
*/
public static ViewHolder getViewHolder(Context context, View convertView,
ViewGroup parent, int layoutId, int position) {
if (convertView == null) {
return new ViewHolder(context, parent, layoutId, position);
} else {
ViewHolder holder = (ViewHolder) convertView.getTag();
// 修改位置變化
holder.mPosition = position;
Log.d(getViewHolder, getViewHolder:position:---> + position);
return holder;
}
}
/**
* find the view by the viewId
*
* @param viewId
* @return
*/
@SuppressWarnings(unchecked)
public T findView(int viewId) {
View view = mViews.get(viewId);
if (view == null) {
view = mConvertView.findViewById(viewId);
// 添加View對象
mViews.put(viewId, view);
}
return (T) view;
}
/**
* get the object of the convertView
*
* @return
*/
public View getConvertView() {
return mConvertView;
}
public int getPosition() {
return mPosition;
}
/**
* 為TextView設置值
*
* @param viewId
* @param text
* @return
*/
public ViewHolder setText(int viewId, String text) {
TextView textView = findView(viewId);
textView.setText(text);
return this;
}
/**
* 為ImageView設置值
*
* @param viewId
* @param resId
* @return
*/
public ViewHolder setImageResource(int viewId, int resId) {
ImageView view = findView(viewId);
view.setImageResource(resId);
return this;
}
// TODO 待寫View設置的輔助的方法
}
(2).CommonAdapter共通類的實現如下:
package com.chengdong.su.baseadapter.util;
import java.util.List;
import android.R.integer;
import android.content.Context;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import com.chengdong.su.baseadapter.R;
public abstract class CommonAdapter extends BaseAdapter {
/** 上下文 */
protected Context mContext;
/** 數據源 */
protected List mData;
private int mLayoutId;
public CommonAdapter(Context mContext, int layoutId) {
super();
this.mContext = mContext;
this.mLayoutId = layoutId;
}
@Override
public int getCount() {
return mData.size();
}
@Override
public T getItem(int position) {
return (T) (mData == null ? 0 : mData.get(position));
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = ViewHolder.getViewHolder(mContext, convertView,
parent, mLayoutId, position);
convert(holder, getItem(position));
return holder.getConvertView();
}
/*** 設置數據 */
public abstract void setData(List data);
/*** 實現給View賦數據的方法 */
public abstract void convert(ViewHolder holder, T item);
}
(3).MyAdapter類的實現:
package com.chengdong.su.baseadapter.adapter;
import java.util.ArrayList;
import java.util.List;
import android.content.Context;
import android.view.View;
import android.widget.CheckBox;
import com.chengdong.su.baseadapter.R;
import com.chengdong.su.baseadapter.bean.Bean;
import com.chengdong.su.baseadapter.util.CommonAdapter;
import com.chengdong.su.baseadapter.util.ViewHolder;
/**
* 通用的ViewHolder
*
* @author scd
*
*/
public class MyAdapter extends CommonAdapter {
private List mList = new ArrayList();
public MyAdapter(Context mContext) {
super(mContext, R.layout.item_listview);
}
@Override
public void setData(List data) {
mData = data;
notifyDataSetChanged();
}
@Override
public void convert(final ViewHolder holder, Bean item) {
// 鏈式編程
holder.setText(R.id.tv_title, item.getTitle())
.setText(R.id.tv_desc, item.getDesc())
.setText(R.id.tv_time, item.getTime())
.setText(R.id.tv_phone, item.getPhone());
// checkBox復用導致內容錯亂的解決方法
final CheckBox checkBox = holder.findView(R.id.cb_select);
// // 默認狀態下都不選中
checkBox.setChecked(false);
if (mList.contains(holder.getPosition())) {
checkBox.setChecked(true);
}
checkBox.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (checkBox.isChecked()) {
mList.add((Integer) holder.getPosition());
} else {
mList.remove((Integer) holder.getPosition());
}
}
});
}
}
(4)MainActivity類的實現如下:
package com.chengdong.su.baseadapter;
import java.util.ArrayList;
import java.util.List;
import com.chengdong.su.baseadapter.adapter.MyAdapter;
import com.chengdong.su.baseadapter.bean.Bean;
import android.app.Activity;
import android.os.Bundle;
import android.widget.ListView;
public class MainActivity extends Activity {
private ListView mListView;
private MyAdapter mAdapter;
private List mData;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initView();
}
private void initView() {
mListView = (ListView) findViewById(R.id.lv_show);
mAdapter = new MyAdapter(MainActivity.this);
initData();
mAdapter.setData(mData);
mListView.setAdapter(mAdapter);
}
private void initData() {
mData = new ArrayList();
Bean bean = new Bean(1, 2, 3, 4);
for (int i = 0; i < 11; i++) {
mData.add(bean);
}
}
}
(5).activity_main類的實現如下:
(6).item_listview類的實現:
說明:
android:descendantFocusability=blocksDescendants
android:focusable=false
是當存在checkbox或者其他控件存在導致item搶占焦點的解決方案:解決item不能被選中點擊。
// checkBox復用導致內容錯亂的解決方法如下:
final CheckBox checkBox = holder.findView(R.id.cb_select);
// // 默認狀態下都不選中
checkBox.setChecked(false);
if (mList.contains(holder.getPosition())) {
checkBox.setChecked(true);
}
checkBox.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (checkBox.isChecked()) {
mList.add((Integer) holder.getPosition());
} else {
mList.remove((Integer) holder.getPosition());
}
}
});
一、簡介現在的Android應用程序中,不可避免的都會使用到圖片,如果每次加載圖片的時候都要從網絡重新拉取,這樣不但很耗費用戶的流量,而且圖片加載的也會很慢,用戶體驗很不
學習到的知識整個事件流向應該是從Activity—->ViewGroup—>View 從上往下調用dispatchTouchEvent
我是完全根據這裡仿制了一個作為備忘,可以點擊這裡查看原始版本代碼如下:1、res/values/attrs.xml 2、具體實現 publ
菜單是用戶界面中最常見的元素之一,使用非常頻繁,在Android中,菜單被分為如下三種,選項菜單(OptionsMenu)、上下文菜單(ContextMenu)和子菜單(