Android SimpleAdapter源码详解

 
  

一直没认真看过android的源码,也不太敢看,稀里糊涂也敲了一年的代码,现在想好好学习了,就把常用的源码都看了一下,小伙伴们来涨姿势吧,有错误的地方,直接指出,我脸厚不怕丢人。来吧。

刚开始学android的时候我经常使用SimpleAdapter,但是后来经常用到的对象实体,SimpleAdapter也就不符合要求了,一直自己继承BaseAdapter,但是有的地方用SimpleAdapter还是比较方便的,一句话就搞定了,也不用写Adapter,所以来悄悄源码吧。
SimpleAdapter的初始化:

 
  
SimpleAdapter sAdapter=new SimpleAdapter(this, mList, R.layout.activity_main,new String[]{"name","pwd"},new int[]{R.id.tv_name,R.id.tv_pwd});
 
  
下面是android的源码,加了详细的注释,不用再多说:
public class SimpleAdapter extends BaseAdapter implements Filterable {

	private int[] mTo; // 指向布局里面控件的id 比如:R.id.btn

	private String[] mFrom; // 数据来源,来自Map里面的key

	private ViewBinder mViewBinder;// 接口类型,里面有个setViewValue方法,用于出现特殊类型控件比如:drawable的时候在外部初始化接口,实现具体方法



	private List<? extends Map<String, ?>> mData;// 用List打包的Map数据源



	private int mResource;// 布局

	private int mDropDownResource;// 不知道干嘛用的,但是估计也是留给外部调用的

	private LayoutInflater mInflater;// 这个大家都知道,LayoutInflater用来载入界面



	private SimpleFilter mFilter;// 过滤器,一般用不到

	private ArrayList<Map<String, ?>> mUnfilteredData;



	// SimpleAdapter初始化,将传进了的参数都赋给本地的对应变量

	public SimpleAdapter(Context context, List<? extends Map<String, ?>> data,

			int resource, String[] from, int[] to) {

		mData = data;

		mResource = mDropDownResource = resource;

		mFrom = from;

		mTo = to;

		mInflater = (LayoutInflater) context

				.getSystemService(Context.LAYOUT_INFLATER_SERVICE);

	}



	/*

	 * ListView 针对每个item,要求 adapter “返回一个视图” (getView),

	 * 也就是说ListView在开始绘制的时候,系统首先调用getCount()函数,根据他的返回值得到ListView的长度,

	 * 然后根据这个长度,调用getView()一行一行的绘制ListView的每一项。如果你的getCount()返回值是0的话,

	 * 列表一行都不会显示,如果返回1,就只显示一行。返回几则显示几行。如果我们有几千几万甚至更多的item要显示怎么办?

	 * 为每个Item创建一个新的View?不可能!!!实际上Android早已经缓存了这些视图,大家可以看下下面这个截图来理解下,

	 * 这个图是解释ListView工作原理的最经典的图了大家可以收藏下,不懂的时候拿来看看,加深理解,

	 * 其实Android中有个叫做Recycler的构件

	 */

	public int getCount() {

		return mData.size();

	}



	public Object getItem(int position) {

		return mData.get(position);

	}



	public long getItemId(int position) {

		return position;

	}



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

		// 调用createViewFromResource来优化内存

		return createViewFromResource(position, convertView, parent, mResource);

	}



	/*

	 * ListView的机制在这里再说一下,当第一个数据来getView的时候convertView肯定是null,

	 * 那么就用mInflater.inflate(resource, parent,

	 * false)给它初始化一个View,但是当一屏滑到底了,第一个item,滑出了屏幕,那么它将

	 * 从底部出来,那时候convertView就不为null,这个方法的好处就是当convertView不为null

	 * 的时候不用加载布局,直接使用convertView, 节省了一步,这也是所说的优化ListView的第一个步骤。

	 */

	private View createViewFromResource(int position, View convertView,

			ViewGroup parent, int resource) {

		View v;

		if (convertView == null) {

			v = mInflater.inflate(resource, parent, false);

		} else {

			v = convertView;

		}

		// 这个方法是核心

		bindView(position, v);



		return v;

	}



	public void setDropDownViewResource(int resource) {

		this.mDropDownResource = resource;

	}



	@Override

	public View getDropDownView(int position, View convertView, ViewGroup parent) {

		return createViewFromResource(position, convertView, parent,

				mDropDownResource);

	}



	// 这个方法的主要功能就是按照 to数组里面控件的顺序挨个赋值,比如new int[]{R.id.tv_name,R.id.tv_pwd}。

	private void bindView(int position, View view) {

		final Map dataSet = mData.get(position);// 找到对应的position位置的map数据

		// 如果没找到跳出

		if (dataSet == null) {

			return;

		}

		/*

		 * 将外部实现的ViewBinder,赋值给本地,SimpleAdapter能够实现:

		 * checkBox,CheckedTextView,TextView

		 * ,ImageView,数据类型也就是Boolean,Integer,Stirng, 所以特殊数据类型的时候才用到ViewBinder

		 * ,如果没用到就不需要外部实现ViewBinder接口和里面的方法

		 */

		final ViewBinder binder = mViewBinder;

		final String[] from = mFrom;

		final int[] to = mTo;

		final int count = to.length;

//view.findViewById(to[i]),循环找控件

		for (int i = 0; i < count; i++) {

			final View v = view.findViewById(to[i]);

			//如果v不为空的话,找到对应的数据,Object类型的data转换为String,因为boolean在map里面纯的也是true或者false

			if (v != null) {

				final Object data = dataSet.get(from[i]);

				String text = data == null ? "" : data.toString();

				if (text == null) {

					text = "";

				}

//标志变量bound,判断外部有没有实现ViewBinder,如果实现了,就执行binder.setViewValue(v, data, text),如果符合特殊条件,就返回true

				boolean bound = false;

				if (binder != null) {

					bound = binder.setViewValue(v, data, text);

				}

//如果满足if (!bound)那么bound就还是false,说明是普通数据

				if (!bound) {

					//查看v是不是Checkable的实例化类型,满足的情况可能是CheckBox,CheckedTextView

					if (v instanceof Checkable) {

						//如果数据类型是boolean,那么就是CheckBox

						if (data instanceof Boolean) {

							((Checkable) v).setChecked((Boolean) data);

							//如果不是CheckBox,那么判断是不是继承TextView的CheckedTextView,是的话赋值,不是就抛出异常

						} else if (v instanceof TextView) {

							setViewText((TextView) v, text);

						} else {

							throw new IllegalStateException(v.getClass()

									.getName()

									+ " should be bound to a Boolean, not a "

									+ (data == null ? "<unknown type>"

											: data.getClass()));

						}

						//如果不是Checkable的实例化类型,判断是不是TextView的实例化类型

					} else if (v instanceof TextView) {

						setViewText((TextView) v, text);

						//都不是以上情况,就判断一下是不是ImageView的实例化类型

					} else if (v instanceof ImageView) {

						//这里只满足数据类型为int也就是R.drawable.ic,和String类型的url,如果想实现直接用drawbale,就要实现ViewBinder

						if (data instanceof Integer) {

							setViewImage((ImageView) v, (Integer) data);

						} else {

							setViewImage((ImageView) v, text);

						}

					} else {

						throw new IllegalStateException(

								v.getClass().getName()

										+ " is not a "

										+ " view that can be bounds by this SimpleAdapter");

					}

				}

			}

		}

	}



	public ViewBinder getViewBinder() {

		return mViewBinder;

	}



	public void setViewBinder(ViewBinder viewBinder) {

		mViewBinder = viewBinder;

	}



	public void setViewImage(ImageView v, int value) {

		v.setImageResource(value);

	}



	public void setViewImage(ImageView v, String value) {

		try {

			v.setImageResource(Integer.parseInt(value));

		} catch (NumberFormatException nfe) {

			v.setImageURI(Uri.parse(value));

		}

	}



	public void setViewText(TextView v, String text) {

		v.setText(text);

	}



	public Filter getFilter() {

		if (mFilter == null) {

			mFilter = new SimpleFilter();

		}

		return mFilter;

	}



	public static interface ViewBinder {

		boolean setViewValue(View view, Object data, String textRepresentation);

	}

//这个不知道干嘛用的,也没用过,好像是过滤数据的

	private class SimpleFilter extends Filter {



		@Override

		protected FilterResults performFiltering(CharSequence prefix) {

			FilterResults results = new FilterResults();



			if (mUnfilteredData == null) {

				mUnfilteredData = new ArrayList<Map<String, ?>>(mData);

			}



			if (prefix == null || prefix.length() == 0) {

				ArrayList<Map<String, ?>> list = mUnfilteredData;

				results.values = list;

				results.count = list.size();

			} else {

				String prefixString = prefix.toString().toLowerCase();



				ArrayList<Map<String, ?>> unfilteredValues = mUnfilteredData;

				int count = unfilteredValues.size();



				ArrayList<Map<String, ?>> newValues = new ArrayList<Map<String, ?>>(

						count);



				for (int i = 0; i < count; i++) {

					Map<String, ?> h = unfilteredValues.get(i);

					if (h != null) {



						int len = mTo.length;



						for (int j = 0; j < len; j++) {

							String str = (String) h.get(mFrom[j]);



							String[] words = str.split(" ");

							int wordCount = words.length;



							for (int k = 0; k < wordCount; k++) {

								String word = words[k];



								if (word.toLowerCase().startsWith(prefixString)) {

									newValues.add(h);

									break;

								}

							}

						}

					}

				}



				results.values = newValues;

				results.count = newValues.size();

			}



			return results;

		}



		@Override

		protected void publishResults(CharSequence constraint,

				FilterResults results) {

			// noinspection unchecked

			mData = (List<Map<String, ?>>) results.values;

			if (results.count > 0) {

				notifyDataSetChanged();

			} else {

				notifyDataSetInvalidated();

			}

		}

	}

}


最近在公司涨了很多姿势,大部分是Java的继承,接口,多态,抽象这些学过但是理解很肤浅的东西,以前自己写软件没什么规范,也没怎么去抽象,所以代码很不上档次,老鸟看到我的代码就笑话我,现在我准备好好学一下,希望自己有一天也能写出很正规的代码。
 
  


                            

你可能感兴趣的:(SimpleAdapter)