在使用ListView和RecycleView之前,我们得先了解适配器的概念。
具体来说,适配器Adapter是一个接口。
官方文档中是这样描述的:
An Adapter object acts as a bridge between an AdapterView and the underlying data for that view. The Adapter provides access to the data items. The Adapter is also responsible for making a android.view.View for each item in the data set.
简单来说,适配器充当着数据与视图之间交互的桥梁,将数据以合适的方式显示出来。其属于MVC模式(数据模式,控制器,视图)的一种具体情况。
getView(int position,View convertView,ViewGroup parent) :
这应该是适配器中最重要的方法了,这个方法会在每个子项被滚动到屏幕内的时候被调用,用于加载视图。
getItem(int positon):
用于获取在数据集中第position位置的实例。
该方法也会在我们设置监听器后用来简单地获取数据。
getItemId(int position)
用于获取在数据集中第position位置的实例对应的ID。
不同getItem的是,某些方法(如onclicklistener的onclick方法)有id这个参数,而这个id参数就是取决于getItemId()这个返回值的。
getCount()
返回一共有几项数据。
我们主要介绍的是ArrayAdapter类的使用,主要通过对该类进行继承重写来实现我们自己的适配器。
1.设计一个布局以显示列表中的每一个子项
这里我简单地用一个图片+文本的形式显示一个子项,在layout下新建一个布局:
2.创建一个中间类以在适配器中加载布局,由于仅有一个图片和一个文本框,所以这里该类比较简单:
public class ItemOne {
private int imageId; //用于存储图片的id值
private String text; //用于存储文本框要显示的内容
public ItemOne(String text,int imageId)
{
this.text = text;
this.imageId = imageId;
}
public String getText()
{
return text;
}
public int getImageId()
{
return imageId;
}
}
3.重点:创建自己的适配器类:
public class MyAdapter extends ArrayAdapter {
private int layoutID;//仅仅用作一个中间变量,用于将构造方法中的ResoruceId(即步骤一中创建的子项布局Id)传递给getView方法
public MyAdapter(Context context, int ResoruceId, List data)
{
super(context,ResoruceId,data); //调用父类的一种构造方法
layoutID = ResoruceId;
}
@Override
public View getView(int position, View convertView, ViewGroup parent)
{
ItemOne itemOne = getItem(position); //获取具体的中间类的实例,
//这里的position其实就是itemOne在具体的List或者Array(构造方法中传入的List)中的索引
View view = LayoutInflater.from(getContext()).inflate(layoutID, parent,false);
//此处是创建具体的子项,调用静态的LayoutInflater.from()从给定的上下文中获取LayoutInflater实例,再调用其inflate方法将具体的子项布局加载进来。
//inflate方法的第一个参数是子项布局的Id,第二个是父布局。第三个是是否给这个View添加父布局,这里我们传入false就行了,因为一旦View有父布局,其就不能添加入ListView之中了。
ImageView imageView = (ImageView) view.findViewById(R.id.Image1);
TextView textView = (TextView) view.findViewById(R.id.Text1);
//从之前加载入的子项的布局中获取具体的微件
imageView.setImageResource(itemOne.getImageId());
textView.setText(itemOne.getText());
//为微件设置图片源和要显示的文本,此处要显示的内容与之前的中间类相关
return view;
}
}
4.在活动中具体加载和使用ListView:
1.首先我们要在主布局中创建一个ListView
2.在主活动中调用并加载ListView :
public class MainActivity extends AppCompatActivity {
private ArrayList myList = new ArrayList<>();
//一个数据列表,用于存储具体的数据
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initMyItem();
ListView listView = (ListView) findViewById(R.id.listview1);
//获取ListView实例
MyAdapter adapter = new MyAdapter(this,R.layout.item_layout,myList);
//创建自定义的适配器
listView.setAdapter(adapter);
//给listView设置适配器
}
private void initMyItem()//初始化数据列表
{
for(int i = 0; i < 20;i++)
{
ItemOne itemOne = new ItemOne("Item "+ i,R.drawable.qq1);
myList.add(itemOne);
}
}
}
在上面的例子中其实ListView的运行效率是很低的,因为getView()方法每次都会将布局加载一遍,一旦数据量过大就会产生卡顿。 我们可以观察getView方法的参数中有一个convertView参数,该参数用于将之前加载好的布局进行缓存。所以我们可以判断convertView参数是否为空来决定是否加载布局以提升性能:
更改适配器中的getView方法
public View getView(int position, View convertView, ViewGroup parent)
{
ItemOne itemOne = getItem(position);
View view;
if(convertView != null)
{
view = convertView;
}else{
view = LayoutInflater.from(getContext()).inflate(layoutID, parent,false);
}
ImageView imageView = (ImageView) view.findViewById(R.id.Image1);
TextView textView = (TextView) view.findViewById(R.id.Text1);
imageView.setImageResource(itemOne.getImageId());
textView.setText(itemOne.getText());
return view;
}
既然我们能够缓存之前的view,那我们还可以借助缓存的view继续对性能进行优化。
public View getView(int position, View convertView, ViewGroup parent)
{
ItemOne itemOne = getItem(position);
View view;
ViewHolder viewHolder = new ViewHolder();
if(convertView != null)
{
view = convertView;
viewHolder = (ViewHolder) view.getTag();//重新获取viewHolder实例
}else{
view = LayoutInflater.from(getContext()).inflate(layoutID,parent,false);
viewHolder.imageView= (ImageView) view.findViewById(R.id.Image1);
viewHolder.textView = (TextView) view.findViewById(R.id.Text1);
view.setTag(viewHolder); //将viewHolder存储在view中
}
viewHolder.imageView.setImageResource(itemOne.getImageId());
viewHolder.textView.setText(itemOne.getText());
return view;
}
class ViewHolder{
ImageView imageView;
TextView textView;
}
}
在这里我们自建了一个内部类ViewHolder用于存储具体的微件,并将其存储在view中,如果view已经被缓存过,则说明其中的微件已经绑定过,所以直接拿出来用就行了。
ListView的鼠标监听事件与Button的监听事件十分相似,不同的点就在于具体实现的接口,调用的方法不同:
修改onCreate方法
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initMyItem();
ListView listView = (ListView) findViewById(R.id.listview1);
MyAdapter adapter = new MyAdapter(this,R.layout.item_layout,myList);
listView.setAdapter(adapter);
listView.setOnItemClickListener(new AdapterView.
OnItemClickListener() {
@Override
public void onItemClick(AdapterView> parent, View view,
int position, long id) {
ItemOne itemOne = myList.get(position);
//获取具体实例
Log.d("MainActivity",itemOne.getText());
}
});
}
这里简单对ListView设置了一个适配器的监听器,每次点击就获取相应实例并且将其text通过日志打印出来.
RecyclerView的用法和ListView十分相似:
1.创建具体的适配器
public class RecycleAdapter extends RecyclerView.Adapter {
private List mList;
static class ViewHolder extends RecyclerView.ViewHolder{
ImageView imageView;
TextView textView;
public ViewHolder(View view)
{
super(view);
imageView = (ImageView) view.findViewById(R.id.Image1);
textView = (TextView) view.findViewById(R.id.Text1);
}
}
public RecycleAdapter(List list)
{
mList = list;
}
@Override
public ViewHolder onCreateViewHolder(ViewGroup parent,int viewType)
{
View view = LayoutInflater.from(parent.getContext())
.inflate(R.layout.item_layout,parent,false);
ViewHolder holder = new ViewHolder(view);
return holder;
}
@Override
public void onBindViewHolder(ViewHolder holder,int position)
{
ItemOne itemOne = mList.get(position);
holder.imageView.setImageResource(itemOne.getImageId());
holder.textView.setText(itemOne.getText());
}
@Override
public int getItemCount(){
return mList.size();
}
}
我们首先新创建一个适配器类继承RecyclerView.Adapter。在适配器中我们创建了一个静态内部类ViewHolder继承RecyclerView.ViewHold,这个类主要是在后面的方法中使用的。
由于继承了RecyclerView.Adapter,该类需要重写三个方法分别用于创建ViewHolder,绑定ViewHolder和返回数据一共有多少项.
2.在主活动中配置适配器:
public class MainActivity extends AppCompatActivity {
private ArrayList myList = new ArrayList<>();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initMyItem();
RecyclerView recyclerView = (RecyclerView)
findViewById(R.id.Re1);
LinearLayoutManager layoutManager = new
LinearLayoutManager(this);
//设置线性布局管理器
recyclerView.setLayoutManager(layoutManager);
RecycleAdapter adapter = new RecycleAdapter(myList);
recyclerView.setAdapter(adapter);
}
private void initMyItem()
{
for(int i = 0; i < 20;i++)
{
ItemOne itemOne = new ItemOne("Item "+ i,R.drawable.qq1);
myList.add(itemOne);
}
}
}
与ListView不同的可能就是RecyclerView还需要额外设置布局方式,这里我们创建的是最简单的线性布局,效果与ListView一样。RecyclerView还有其他的布局模式,我们可以自行去体验。