原创 继承 BaseAdapter 自定义 Adapter 来实现 ListView

2021-6-2 17:14 1543 15 15 分类: 软件与OS 文集: android studio

其实 SimpleAdapter 适配器已经能够满足很多情况了,但是由于它的一些缺陷,实际上在开发中使用最多、最广泛的还是通过继承 BaseAdapter 自定义 Adapter 的方式来实现ListView SimpleAdapterArrayAdapter也是继承自 BaseAdapter 的,只是两种已经实现好的 BaseAdapter而已。

  SimpleAdapter ArrayAdapter 一样,自定义 Adapter 需要至少实现 BaseAdapter getCount()getltem(int position)getltemld(int position)以及 getView(final int positionView convertViewViewGroup parent) 4 个方法。下面通过实例说明这 4 个方法的作用。实例将在 res/layout 文件夹下创建一个item list.xml布局文件,通过在这个 item 中加入一个 button 按钮来实现当点击该按钮时删除 item 的功能。item 的布局文件如下

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. android:orientation="horizontal">
  6. <ImageView
  7. android:id="@+id/itemImage"
  8. android:layout_width="0dp"
  9. android:layout_height="wrap_content"
  10. android:layout_weight="1"/>
  11. <TextView
  12. android:id="@+id/itemText"
  13. android:layout_width="0dp"
  14. android:layout_height="wrap_content"
  15. android:layout_weight="1"
  16. android:textSize="20sp"/>
  17. <Button
  18. android:id="@+id/itemButton"
  19. android:layout_width="wrap_content"
  20. android:layout_height="wrap_content"
  21. android:text="删除" />
  22. </LinearLayout>

MainActivity对应的布局文件中加入ListView,activity_main.xml代码如下:

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. xmlns:tools="http://schemas.android.com/tools"
  4. android:layout_width="match_parent"
  5. android:layout_height="match_parent"
  6. android:orientation="horizontal"
  7. tools:context=".MainActivity">
  8. <ListView
  9. android:id="@+id/listView"
  10. android:layout_width="match_parent"
  11. android:layout_height="wrap_content"/>
  12. </LinearLayout>

本实例中写一个自定义的适配器类继承BaseAdapter,代码如下:

  1. package com.rfstar.listviewtest03;
  2. import android.content.Context;
  3. import android.view.LayoutInflater;
  4. import android.view.View;
  5. import android.view.ViewGroup;
  6. import android.widget.BaseAdapter;
  7. import android.widget.Button;
  8. import android.widget.ImageView;
  9. import android.widget.TextView;
  10. import android.widget.Toast;
  11. import java.util.List;
  12. import java.util.Map;
  13. public class MyAdapter extends BaseAdapter {
  14. private Context context;
  15. private List<Map<String,Object>>dataList;
  16. public MyAdapter(Context context,List<Map<String,Object>> dataList){
  17. this.context=context;
  18. this.dataList=dataList;
  19. }
  20. @Override
  21. public int getCount() {
  22. return dataList.size();
  23. }
  24. @Override
  25. public Object getItem(int position) {
  26. return dataList.get(position);
  27. }
  28. @Override
  29. public long getItemId(int position) {
  30. return position;
  31. }
  32. @Override
  33. public View getView(final int position, View convertview, ViewGroup viewGroup) {
  34. if (convertview == null) {
  35. convertview = LayoutInflater.from(context).inflate(R.layout.item_layout, null);
  36. }
  37. ImageView image=(ImageView)convertview.findViewById(R.id.itemImage);
  38. TextView textView=(TextView)convertview.findViewById(R.id.itemText);
  39. Button delet=(Button)convertview.findViewById(R.id.itemButton);
  40. Map<String,Object> map=dataList.get(position);
  41. image.setImageResource((Integer)map.get("image"));
  42. textView.setText((String)map.get("name"));
  43. delet.setOnClickListener(new View.OnClickListener() {
  44. @Override
  45. public void onClick(View view) {
  46. Toast.makeText(context,"删除了第"+position+"行",Toast.LENGTH_LONG).show();
  47. dataList.remove(position);
  48. notifyDataSetChanged();
  49. }
  50. });
  51. return convertview;
  52. }
  53. }

在这个自定义的适配器类中创建了一个用于接收数据集以及上下文对象的构造方法,并实现了BaseAdapter 4个方法。在这些方法中,getCount()、getItemint position)、getltemldint position)通过代码甚至方法名就能够很容易地理解它们的意义,这里就不做讲解了。

在一个完整的ListView 第一次出现时,每个 item 都是空的,会调用 getViewO方法去创建并返回一个item(一个 View 对象)。这个过程就像代码中显示的,如果convertView 是空的,就通过LayoutInflaterinflate()方法获取这个item 布局对象,并将其赋值给convertView 对象 InflateO)方法与 findViewByIdO方法有些相似之处,不过一个是用来获取布局文件的对象,一个是用来获取控件的对象。当 converView 对象被赋值之后,通过 convertView 对象调用

findViewById()方法来获取各个控件,并对控件进行一系列操作。此处对 Button 按钮添加了一个点击事件,当点击按钮时,将执行dataList.removeposition)和 notifyDataSetChanged()两个方法。在 ListView 中,想要删除某条 item,需要执行的就是这两个方法,第一个方法是从数据集中将对应item 的数据删除,第二个方法是用来观测数据集变化的,当数据集发生变化时,将重新执行 getView()方法,刷新界面,展示出删除该条 item 后的界面。

如果我们有大量的数据需要显示,每个 item 都去 getView中重复执行创建新的view 的动作吗? 答案是并不,系统会根据一个屏幕能够显示的 item 数量来执行创建 view 的动作。

当完成自定义Adapter的工作之后,下面我们看下MainActivity中的代码:

  1. package com.rfstar.listviewtest03;
  2. import androidx.appcompat.app.AppCompatActivity;
  3. import android.os.Bundle;
  4. import android.widget.ListView;
  5. import android.widget.SimpleAdapter;
  6. import java.util.ArrayList;
  7. import java.util.HashMap;
  8. import java.util.List;
  9. import java.util.Map;
  10. public class MainActivity extends AppCompatActivity {
  11. private ListView listView;
  12. private List<Map<String,Object>> dataList=new ArrayList<Map<String, Object>>();
  13. private int[] itemIdArray=new int[]{R.id.itemText,R.id.itemImage};
  14. private String[] dataKeyArray=new String[]{"name","image"};
  15. @Override
  16. protected void onCreate(Bundle savedInstanceState) {
  17. super.onCreate(savedInstanceState);
  18. setContentView(R.layout.activity_main);
  19. initData();
  20. initView();
  21. }
  22. private void initView() {
  23. listView=(ListView)findViewById(R.id.listView);
  24. MyAdapter myAdapter=new MyAdapter(this,dataList);
  25. listView.setAdapter(myAdapter);
  26. }
  27. private void initData() {
  28. Map<String,Object>map;
  29. for(int i=0;i<10;i++)
  30. {
  31. map=new HashMap<String,Object>();
  32. map.put("name","大鸟科创空间"+i);
  33. map.put("image",R.mipmap.ic_launcher);
  34. dataList.add(map);
  35. }
  36. }
  37. }

运行应用,并尝试点击"删除"按钮,效果如图所示。

通过实例发现这种自定义的适配器确实实现了点击"删除"按钮就能删除一条 item 的功能。这也说明了自定义的适配器要比前两种适配器强大得多。

继续分析实例,细心的读者会发现,按照上述代码以及我们之前描述的 getView()方法的原理会出现一种非常消耗内存的状况每次删除一条记录或者上下滑动时会频繁地执行调用 findViewBy()方法去获取控件对象。所以有必要对上述代码进行改造。在开发过程发现最被开发者接受的一种方式是在适配器中添加一个静态内部类来保存 item 的控件对象,在 convertView 为空

时,使用converViewsetTag()方法来保存这个类对象。当convertView不为空时就直接使用 getTag()方法获取这个静态类的对象,然后依靠它获取 item 的控件对象。改动后,适配器类的代码如下:

  1. package com.rfstar.listviewtest03;
  2. import android.content.Context;
  3. import android.view.LayoutInflater;
  4. import android.view.View;
  5. import android.view.ViewGroup;
  6. import android.widget.BaseAdapter;
  7. import android.widget.Button;
  8. import android.widget.ImageView;
  9. import android.widget.TextView;
  10. import android.widget.Toast;
  11. import java.util.List;
  12. import java.util.Map;
  13. public class MyAdapter extends BaseAdapter {
  14. private Context context;
  15. private List<Map<String,Object>>dataList;
  16. public MyAdapter(Context context,List<Map<String,Object>> dataList){
  17. this.context=context;
  18. this.dataList=dataList;
  19. }
  20. @Override
  21. public int getCount() {
  22. return dataList.size();
  23. }
  24. @Override
  25. public Object getItem(int i) {
  26. return dataList.get(i);
  27. }
  28. @Override
  29. public long getItemId(int i) {
  30. return i;
  31. }
  32. static class ViewHolder{ //定义静态类
  33. TextView textView;
  34. Button button;
  35. ImageView imageView;
  36. }
  37. @Override
  38. public View getView(final int position, View convertview, ViewGroup viewGroup) {
  39. ViewHolder holder=null;
  40. if (convertview == null) {
  41. holder=new ViewHolder();//创建静态类对象
  42. convertview = LayoutInflater.from(context).inflate(R.layout.item_layout, null);
  43. holder.textView=(TextView)convertview.findViewById(R.id.itemText);
  44. holder.button=(Button)convertview.findViewById(R.id.itemButton);
  45. holder.imageView=(ImageView)convertview.findViewById(R.id.itemImage);
  46. convertview.setTag(holder);//保存静态类对象
  47. }else
  48. {
  49. holder=(ViewHolder)convertview.getTag();//获取静态类对象
  50. }
  51. Map<String,Object> map=dataList.get(position);
  52. holder.imageView.setImageResource((Integer)map.get("image"));
  53. holder.textView.setText((String)map.get("name"));
  54. holder.button.setOnClickListener(new View.OnClickListener() {
  55. @Override
  56. public void onClick(View view) {
  57. Toast.makeText(context,"删除了第"+position+"行",Toast.LENGTH_LONG).show();
  58. dataList.remove(position);
  59. notifyDataSetChanged();
  60. }
  61. });
  62. return convertview;
  63. }
  64. }

经过改动后,通过分析代码逻辑或者打印 log 的方式可以很明确地得知,确实不会重复去获取item类的控件对象了,而是重复使用已经保存好的 viewHolder 对象。上述改动后的适配器就是在开发实践中得出的最佳自定义适配器,希望读者能够掌握。

android studio工具及手机模拟器以及更多工程源代码下载请前往微信公众号:大鸟科创空间,回复:android studio即可获取。

作者: 大鸟科创空间, 来源:面包板社区

链接: https://mbb.eet-china.com/blog/uid-me-3949041.html

版权声明:本文为博主原创,未经本人允许,禁止转载!

文章评论0条评论)

登录后参与讨论
我要评论
0
15
关闭 站长推荐上一条 /2 下一条