Android中Adapter数据适配器

<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;
	
	@Override
	protected 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);
	}

	@Override
	public 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()本身,即this
			this.myInflater=LayoutInflater.from(context);
		}
		
		@Override
		public int getCount() {
			//在此适配器中所代表的数据集中的条目数
			return listMap.size();
		}

		@Override
		public Object getItem(int position) {
			//获取数据集中与指定索引对应的数据项
			return position;
		}

		@Override
		public long getItemId(int position) {
			//获取在列表中与指定索引对应的行id
			return position;
		}

		//convertView就是每一Item在Recyler之前的布局视图
		@Override
		public View getView(int position, View convertView, ViewGroup parent) {
			ViewHolder holder=null;
			if(convertView==null){
				//如果缓存convertView为空,则需要创建View
				holder=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里,以便后面方便取出Tag
				convertView.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() {
				@Override
				public void onClick(View v) {
					showInfo();                
				}
			});
			return convertView;
		}
	}
	
	/**
	* listview中点击按键弹出对话框
	*/
	public void showInfo(){
		new AlertDialog.Builder(this).setTitle("我的listview").setMessage("介绍...").setPositiveButton("确定", new DialogInterface.OnClickListener() {
			@Override
			public 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
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值