接上文

          上一篇文章主要讨论了Android中Adapter的作用以及与设计模式中适配器模式的共同点,并且给出了

   Android中Adapter的体系结构。虽然在上一篇文章中我们明白了Adapter是数据源和AdapterView之间的

   桥梁,完成AdapterView和数据源“接口”的“适配”。但是Adapter到底是如何工作的,我们还是不甚明了,

   因此本篇文章就是要深入理解Adapter的工作过程,小编也是机器人菜鸟。把自己的感想写下来大家一起

   学习。

          Android中的适配器各式各样,AdapterView也有很多,这里我们选用ListView和BaseAdapter来理解

   其工作原理。使用它们是基于以下几点考虑的:1、ListView在开发中使用较多,应用场景十分广泛。

   2、通过资料的查询了解到BaseAdapter能够实现更变化的布局,并且涉及到ListView的优化问题。好了

   闲话不多说了,来看看具体的例子吧!

         activity_main.xml

                      
           list_item.xml

                                              
           MainActivity.java

package com.example.listview_baseadapter;  import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map;  import android.app.Activity; import android.content.Context; import android.os.Bundle; import android.view.LayoutInflater; import android.view.Menu; import android.view.View; import android.view.ViewGroup; import android.widget.BaseAdapter; import android.widget.ImageView; import android.widget.ListView; import android.widget.TextView;  public class MainActivity extends Activity {  	private ListView listView = null; 	private List> data = null;  	@Override 	protected void onCreate(Bundle savedInstanceState) { 		super.onCreate(savedInstanceState); 		setContentView(R.layout.activity_main); 		listView = (ListView) findViewById(R.id.list);// 取得控件         data = getData();//获取数据         MyAdapter adapter = new MyAdapter(this);         listView.setAdapter(adapter); 	}  	@Override 	public boolean onCreateOptionsMenu(Menu menu) { 		// Inflate the menu; this adds items to the action bar if it is present. 		getMenuInflater().inflate(R.menu.main, menu); 		return true; 	}  	// 得到数据 	public List> getData() { 		List> list = new ArrayList>();  		 Map map; 	        for(int i=0;i<10;i++) 	        { 	            map = new HashMap(); 	            map.put("img", R.drawable.bgs); 	            map.put("title", "Coder");             map.put("content", "简单Coding,快乐生活~~~~"); 	            list.add(map); 	        } 	        return list; 	}  	// ViewHolder静态类 	static class ViewHolder { 		public ImageView img; 		public TextView title; 		public TextView content; 	}  	public class MyAdapter extends BaseAdapter {  		private LayoutInflater mInflater = null;  		private MyAdapter(Context context) { 			// 根据context上下文加载布局 			this.mInflater = LayoutInflater.from(context); 		}  		@Override 		public int getCount() { 			// 在此适配器中所代表的数据集中的条目数 			return data.size(); 		}  		@Override 		public Object getItem(int position) { 			 			// 获取数据集中与指定索引对应的数据项 			return position; 		}  		@Override 		public long getItemId(int position) { 		 			// 获取在列表中与指定索引对应的行id 			return position; 		}  		// 获取一个在数据集中指定索引的视图来显示数据 		@Override 		public View getView(int position, View convertView, ViewGroup parent) { 			ViewHolder holder = null; 			// 如果缓存convertView为空,则需要创建View 			if (convertView == null) { 				holder = new ViewHolder(); 				// 根据自定义的Item布局加载布局 				convertView = mInflater.inflate(R.layout.list_item, null); 				holder.img = (ImageView) convertView.findViewById(R.id.img); 				holder.title = (TextView) convertView.findViewById(R.id.title); 				holder.content = (TextView) convertView.findViewById(R.id.content); 				// 将设置好的布局保存到缓存中,并将其设置在Tag里,以便后面方便取出Tag 				convertView.setTag(holder); 			} else { 				holder = (ViewHolder) convertView.getTag(); 			} 			holder.img.setBackgroundResource((Integer) data.get(position).get( 					"img")); 			holder.title.setText((String) data.get(position).get("title")); 			holder.content.setText((String) data.get(position).get("content"));  			return convertView; 		}  	}  } 
                 直接上效果图:

               需要提及的是上述的代码的实现是考虑到了ListView的优化的,撇开对ListView的优化问题。 MyAdapter类中

       的getView()是最重要的,这里也是影响ListView效率的关键。

             对于getView()方法的处理一般有三种方式:

                      1、没有任何处理,这种方式只能在数据量少的情况下使用,所以不建议使用此方式。

public View getView(int position, View convertView, ViewGroup parent) { 			View item = mInflater.inflate(R.layout.list_item, null);             ImageView img = (ImageView)item.findViewById(R.id.img);              TextView title = (TextView)item.findViewById(R.id.title);             TextView content = (TextView)item.findViewById(R.id.content);             img.setImageResource(R.drawable.bgs);             title.setText("Coding");             content.setText("简单编程,快乐生活~~~");             return item; }
                     2、通过缓存converView优化,这种方式可以盘判断缓存中不存在View才创建,如有则利用缓存中的View

           提升了性能。

public View getView(int position, View convertView, ViewGroup parent) { 			if(convertView == null)             {                 convertView = mInflater.inflate(R.layout.list_item, null);             }             ImageView img = (ImageView)convertView.findViewById(R.id.img);              TextView title = (TextView)convertView.findViewById(R.id.title);             TextView content = (TextView)convertView.findViewById(R.id.content);             img.setImageResource(R.drawable.bgs);             title.setText("Coding");             content.setText("简单编程,快乐生活~~~");             return convertView; }
                     3、第三种方式即是,上述代码实现的部分:通过convertView+ViewHolder来实现,使用ViewHolder

             这个静态类的好处是缓存了显示数据的View,加快了UI的响应速度。

                     当我们判断convertView == null,若为空,就会根据设计好的布局文件布局,并未convertView赋值,

              并且生成一个viewHolder来绑定convertView的各个View控件。再用convertViewsetTagviewHolder

              设置到Tag中,以便系统第二次绘制ListView时从Tag中取出。

                      因此,建议使用第三种方式来优化ListView!
             相关参考:

             http://www.cnblogs.com/over140/archive/2011/03/23/1991100.html

             程序完整源代码:

            http://download.csdn.net/detail/kiritor/5134811