Android中Adapter数据适配器

来源:互联网 发布:期货研究员 知乎 编辑:程序博客网 时间:2024/05/19 14:37

<span style="font-family: Arial, Helvetica, sans-serif; font-size: 12px; background-color: rgb(255, 255, 255);">          今天用</span><span style="font-family: Arial, Helvetica, sans-serif; font-size: 14px; background-color: rgb(255, 255, 255);">Android中BaseAdapter做了一个数据集合的展示,自己把自己给坑了。所以自己很有必要搞清楚Adapter的原理。</span>

       对于Adapter的原理有一个很形象的比喻:现在有4瓶水《农夫山泉、康师傅、冰红茶、脉动》,我们知道这个4个瓶子的形状都不一样,尤其是瓶口不一样。现在需要我们向一个大的康师傅的瓶子里倒水,要求水不泼洒。

       肿么办?

       于是漏斗解决了这个问题。那么这里的漏斗就像是Adapter,而四种不一样的瓶子就如Android中widget数据集控件《ListView、GridView、Gallery、Spinner...》现在要将数据投入到这个里面,那么借助Adapter肿么实现呢?

       Android的官方提供了很多数据适配器,将数据绑定到指定的View试图中

       这里只介绍BaseAdapter

       --------------------

       使用数据适配器实现数据列表中增加图片和两行文本以及一个按钮。下面是代码

      Main.xml  

<?xml version="1.0" encoding="UTF-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"     android:layout_width="fill_parent"     android:layout_height="fill_parent"     android:orientation="vertical">      <ListView         android:id="@+id/listView"         android:layout_width="fill_parent"         android:layout_height="wrap_content"         android:fastScrollEnabled="true" />  </LinearLayout> 
      Activity

     

package com.lol.huixin.control;import android.os.Bundle;import android.util.Log;import java.util.ArrayList;import java.util.HashMap;import java.util.List;import java.util.Map;import android.app.Activity;import android.app.AlertDialog;import android.app.ListActivity;import android.content.Context;import android.content.DialogInterface;import android.view.LayoutInflater;import android.view.Menu;import android.view.View;import android.view.ViewGroup;import android.widget.BaseAdapter;import android.widget.Button;import android.widget.ImageView;import android.widget.ListView;import android.widget.TextView;public class BaseAdapterActivity extends Activity {private ListView listView;private List<Map<String,Object>> listMap;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_base_adapter);listView=(ListView)findViewById(R.id.listView);listMap=getData();MyAdapter adapter=new MyAdapter(getApplicationContext());listView.setAdapter(adapter);}@Overridepublic boolean onCreateOptionsMenu(Menu menu) {// Inflate the menu; this adds items to the action bar if it is present.getMenuInflater().inflate(R.menu.base_adapter, menu);return true;}private List<Map<String, Object>> getData() {List<Map<String, Object>> list = new ArrayList<Map<String, Object>>();Map<String, Object> map = new HashMap<String, Object>();map.put("title", "G1");map.put("info", "google 1");map.put("img", R.drawable.a1);list.add(map);map = new HashMap<String, Object>();map.put("title", "G2");map.put("info", "google 2");map.put("img", R.drawable.a2);list.add(map);map = new HashMap<String, Object>();map.put("title", "G3");map.put("info", "google 3");map.put("img", R.drawable.a3);list.add(map);return list;}public static class ViewHolder{public ImageView img;public TextView title;public TextView info;public Button viewBtn;}public class MyAdapter extends BaseAdapter{//LayoutInflater是用来寻找layout下xml布局文件,并且实例化// findViewById()是找具体xml下具体的widget控件(Button TextView)public LayoutInflater myInflater;public MyAdapter(Context context){////根据context上下文加载布局,这里的是getApplicationContext()本身,即thisthis.myInflater=LayoutInflater.from(context);}@Overridepublic int getCount() {//在此适配器中所代表的数据集中的条目数return listMap.size();}@Overridepublic Object getItem(int position) {//获取数据集中与指定索引对应的数据项return position;}@Overridepublic long getItemId(int position) {//获取在列表中与指定索引对应的行idreturn position;}//convertView就是每一Item在Recyler之前的布局视图@Overridepublic View getView(int position, View convertView, ViewGroup parent) {ViewHolder holder=null;if(convertView==null){//如果缓存convertView为空,则需要创建Viewholder=new ViewHolder();//这句话很关键convertView=myInflater.inflate(R.layout.list_items, null);holder.img=(ImageView)convertView.findViewById(R.id.img);holder.title=(TextView)convertView.findViewById(R.id.title);holder.info=(TextView)convertView.findViewById(R.id.info);holder.viewBtn=(Button)convertView.findViewById(R.id.view_btn);//将设置好的布局保存到缓存中,并将其设置在Tag里,以便后面方便取出TagconvertView.setTag(holder);}else{holder=(ViewHolder)convertView.getTag();}Log.i("positon",String.valueOf(position)+"\t"+listMap.get(position).get("title"));holder.img.setBackgroundResource((Integer)listMap.get(position).get("img"));holder.title.setText((String)listMap.get(position).get("title"));holder.info.setText((String)listMap.get(position).get("info"));holder.viewBtn.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {showInfo();                }});return convertView;}}/*** listview中点击按键弹出对话框*/public void showInfo(){new AlertDialog.Builder(this).setTitle("我的listview").setMessage("介绍...").setPositiveButton("确定", new DialogInterface.OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {}}).show();}}
    list_items.xml

    

<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:orientation="horizontal" android:layout_width="fill_parent"android:layout_height="fill_parent"><ImageView android:id="@+id/img" android:layout_width="wrap_content"android:layout_height="wrap_content" android:layout_margin="5px"/><LinearLayout android:orientation="vertical"android:layout_width="wrap_content" android:layout_height="wrap_content"><TextView android:id="@+id/title" android:layout_width="wrap_content"android:layout_height="wrap_content" android:textColor="#FFFFFFFF"android:textSize="22px" /><TextView android:id="@+id/info" android:layout_width="wrap_content"android:layout_height="wrap_content" android:textColor="#FFFFFFFF"android:textSize="13px" /></LinearLayout><Button android:id="@+id/view_btn"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="@string/s_view_btn"android:layout_gravity="bottom|right" /></LinearLayout>
     效果是这样:

      

      我们来想为什么,要这么写?

      虽然每一行代码都有注释,但是为啥这么一个简单的功能,要写这么多代码,脑壳有坑麽?

      1.首先BaseAdapter这个适配器中的getView()方法中的是遍历每一个listMap中每一个items,也就需要我们在每一个item选项中创建图片、文本、按钮试图控件,需要独立一个试图集来存放这些东西,为啥呢,因为getView()方法是一个伪迭代器,他一直在遍历集合,如果我们每次都用findViewById方法去寻找试图组件,肯定是损耗内存。

      2.一般来讲当我们寻找通过Activity创建的xml时,只需要通过findViewById,但是如果我们需要寻找独立创建的xml文件时,特别是在每一个遍历集合Item中寻找时,那么此时就需要判断缓存中是否存在我们寻找的xml,不存在就创建

      3.为啥需要tag,一般来讲我们使用listView试图显示数据时,数据量很大,就涉及到分页翻取,在Android中,我不能每次翻页时,都从数据库中查询,那么会很占资源,而在Adapter适配器中的convertView中恰恰可以存储这个对象,我们通过postion可以了解到,所以这里使用tag可以快速定位我们以前存在缓存中的数据列表

      

0 0
原创粉丝点击