赞
踩
1.概念
众所众知,Adapter是用于连接后端数据和前端显示的适配器接口,是数据data和UI(View)之间一个重要的纽带。在常见的View(ListView,GridView)等地方都需要用到Adapter。如下图直观的表达了Data、Adapter、View三者的关系:
Adapter接口的组成中
其中最常用的有以下几个
BaseAdapter是所有Adapter类的父类,所有的Adapter类的实现都是基于BaseAdapter的基础上的
ArrayAdapter支持泛型操作,最为简单,但是只能展示一行字。
SimpleAdapter虽然名称为simple,但是用法功能还是很强大的,基本上我们在敲代码时都要与这个打交道
2 缓存原理
试想一个场景:若把所有数据集合的信息都加载到ListView上显示,若 ListView要为每个数据都创建一个视图,那么会占用非常多的内存
为了节省空间和时间,ListView不会为每一个数据创建一个视图,而是采用了Recycler组件,用于回收 & 复用 View
当屏幕需显示x个Item时,那么ListView会创建 x+1个视图;当第1个Item离开屏幕时,此Item的View被回收至缓存,入屏的Item的View会优先从该缓存中获取
注:
只有Item完全离开屏幕后才可复用,这也是为什么ListView要创建比屏幕需显示视图多1个的原因:缓冲 显示视图
即:第1个Item离开屏幕是有过程的,会有1个 第1个Item的下半部分 & 第8个Item上半部分同时在屏幕中显示的状态,此时仍无法使用缓存的View,只能继续用新创建的视图View
实例演示
设:屏幕只能显示5个Item,那么ListView只会创建(5+1)个Item的视图;当第1个Item完全离开屏幕后才会回收至缓存从而复用(用于显示第7个Item)
3.应用案例
1)ArrayAdapter
ListView仅作为容器(列表),用于装载 & 显示数据(即 列表项Item)
而容器内的具体数据(列表项Item)则是由 适配器(Adapter)提供
案例一
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
String[] str = {"1","1","1","1","1"};
//创建ArrayAdapter
ArrayAdapter <String> adapter = new ArrayAdapter<String(this,android.R.layout.simple_expandable_list_item_1,str);
//获取listview 对象 list 与xml 中的list_test绑定 并用setAdapter显示
ListView list = (ListView)findViewById(R.id.list_test);//主页面的视图
list.setAdapter(adapter);
}
ArrayAdapter(Context context, int textViewResourceId, List objects)来装配数据,要装配这些数据就需要一个连接ListView视图对象和数组数据的适配器来两者的适配工作,ArrayAdapter的构造需要三个参数,依次为this,布局文件(注意这里的布局文件描述的是列表的每一行的布局,android.R.layout.simple_list_item_1是系统定义好的布局文件只显示一行文字,数据源(一个List集合)。同时用setAdapter()完成适配的最后工作。效果图如下:
**2)*SimpleAdapter*
** simpleAdapter,可以定义各种各样的布局出来,可以放上ImageView(图片),还可以放上Button(按钮),CheckBox(复选框)等等。如下,做出了一个类似聊天目录的界面
实现效果:
mainactivity.java
public class MainActivity extends Activity { private String[] animals = new String[]{"lion", "cat", "dog", "elephant", "monkey", "tiger"}; private int[] imageIds = new int[]{R.drawable.lion, R.drawable.cat, R.drawable.dog, R.drawable.elephant, R.drawable.monkey, R.drawable.tiger}; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); List<Map<String, Object>> listItems = new ArrayList<Map<String, Object>>(); for (int i = 0; i < animals.length; i++) { Map<String, Object> showitem = new HashMap<>(); showitem.put("tximage", imageIds[i]); showitem.put("name", animals[i]); listItems.add(showitem); SimpleAdapter myAdapter = new SimpleAdapter(getApplicationContext(), listItems, R.layout.list_item, new String[]{"tximage", "name"}, new int[]{R.id.imgtou, R.id.name}); ListView listView = (ListView) findViewById(R.id.list_test); listView.setAdapter(myAdapter); //设置setOnItemClickListener listView.setOnItemClickListener(new AdapterView.OnItemClickListener(){ @Override public void onItemClick(AdapterView<?> parent, View view, int position, long id){ Toast.makeText(getApplicationContext(), animals[position], Toast.LENGTH_LONG).show(); } }); listView.setOnItemLongClickListener(new AdapterView.OnItemLongClickListener() { @Override public boolean onItemLongClick(AdapterView<?> parent, View view, int position, long id) { Toast.makeText(getApplicationContext(),animals[position]+"longClicked",Toast.LENGTH_LONG).show(); return true; } }); } } }
activity.xml 用于主页面的布局
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity">
<ListView
android:id="@+id/list_test"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
</LinearLayout>
layout.xml 用于定义listItem 中的布局文件
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="horizontal"> <!-- 定义一个竖直方向的LinearLayout,显示呢称 --> <LinearLayout android:layout_width="320dp" android:layout_height="wrap_content" android:orientation="vertical"> <!-- 定义一个用于显示头像的ImageView --> <TextView android:id="@+id/name" android:layout_width="wrap_content" android:layout_height="wrap_content" android:paddingLeft="8dp" android:textColor="#1D1D1C" android:textSize="20sp" /> <TextView android:id="@+id/says" android:layout_width="wrap_content" android:layout_height="wrap_content" android:paddingLeft="8px" android:textColor="#B4B4B9" android:textSize="14sp" /> </LinearLayout> <ImageView android:id="@+id/imgtou" android:layout_width="40dp" android:layout_height="40dp" android:layout_marginTop="10dp" /> </LinearLayout>
**使用simpleAdapter的数据用一般都是HashMap构成的List,list的每一节对应ListView的每一行。**HashMap的每个键值数据映射到布局文件中对应id的组件上,通过findViewbyId。我们自己定义一个布局list_Item.xml用于定义list_Item的布局。SimpleAdapter参数一次是:this,布局文件),HashMap的 title 和 info,img。布局文件的组件id,title,info,img。布局文件的各组件分别映射到HashMap的各元素上,完成适配。
总结一下就是:
1. 在xml文件布局上实现ListView
2. 根据实际需求定制列表项:实现ListView每行的xml布局(即item布局)
3. 定义一个HashMap构成的列表以键值对的方式存放数据
4. 构造SimpleAdapter对象,设置适配器
5. 将LsitView绑定到SimpleAdapter上
3)BaseAdapter(仅做简单描述)
定义
可自定义ListView,通用用于被扩展。扩展BaseAdapter可以对各个列表项进行最大程度的定制
使用步骤:与simpleAdapter大同小异
1.定义主xml布局
2.根据需要定义ListView每行所实现的xml布局
3.定义一个Adapter类继承BaseAdapter,重写里面的方法。
4.定义一个HashMap构成的列表,将数据以键值对的方式存放在里面。
5.构造Adapter对象,设置适配器。
6.将ListView绑定到Adapter上。
先定义一个Adapter类继承BaseAdapter,并重写里面的方法
使用BaseAdapter必须写一个类继承它,同时BaseAdapter是一个抽象类,继承它必须实现它的方法。
这里主要讲一下BaseAdapter里必须要重写的4个方法
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import java.util.List;
public abstract class BaseListAdapter<T> extends BaseAdapter {
private List<T> mList;
private final Context mContext;
public BaseListAdapter(Context context) { this.mContext = context; } public BaseListAdapter(Context context, List<T> list) { mContext = context; mList = list; } public Context getContext() { return mContext; } @Override public int getCount() { return mList != null ? mList.size() : 0; } @Override public T getItem(int position) { return mList != null ? mList.get(position) : null; } @Override public long getItemId(int position) { return position; } public void setData(List<T> list) { mList = list; notifyDataSetChanged(); } public List<T> getList() { return mList; } @Override public View getView(int position, View convertView, ViewGroup parent) { if (convertView == null) { convertView = onNewItemView(LayoutInflater.from(mContext), parent); } onBindItemView(convertView, getItem(position), position, parent); return convertView; } protected abstract View onNewItemView(LayoutInflater inflater, ViewGroup parent); protected abstract void onBindItemView(View viewItem, T data, int position, ViewGroup parent); }
参考链接:Carson带你学Android:全面解析列表ListView与AdapterView(https://blog.csdn.net/carson_ho/article/details/51472640)
本文作者:不知名抑术家
本文链接:https://blog.csdn.net/weixin_46621811/article/details/121895894
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。