zoukankan      html  css  js  c++  java
  • Android中ListView使用总结

    LisView介绍:

    (一)、ListView 概念:

    ListView是Android中最重要的组件之一,几乎每个Android应用中都会使用ListView。它以垂直列表的方式列出所需的列表项。
    java.lang.Object
       ↳ android.view.View
        ↳ android.view.ViewGroup
          ↳ android.widget.AdapterView
            ↳ android.widget.AbsListView
              ↳ android.widget.ListView

    (二)、ListView的两个职责:

    • 将数据填充到布局;
    • 处理用户的选择点击等操作。

    (三)、列表的显示需要三个元素:

    • 1.ListVeiw:用来展示列表的View;
    • 2.适配器: 用来把数据映射到ListView上的中介;
    • 3.数据源: 具体的将被映射的字符串,图片,或者基本组件。

    (四)、什么是适配器?

    适配器是一个连接数据和AdapterView的桥梁,通过它能有效地实现数据与AdapterView的分离设置,使AdapterView与数据的绑定更加简便,修改更加方便。将数据源的数据适配到ListView中的常用适配器有:ArrayAdapter、SimpleAdapter 和 SimpleCursorAdapter。

    • ArrayAdapter最为简单,只能展示一行字;
    • SimpleAdapter有最好的扩充性,可以自定义各种各样的布局,除了文本外,还可以放ImageView(图片)、Button(按钮)、CheckBox(复选框)等等;
    • SimpleCursorAdapter可以认为是SimpleAdapter对数据库的简单结合,可以方便地把数据库的内容以列表的形式展示出来。
    • 但是实际工作中,常用自定义适配器。即继承于BaseAdapter的自定义适配器类。

    (五)、ListView的常用UI属性:

    • android:divider
    • android:dividerHeight
    • android:entries
    • android:footerDividersEnabled
    • android:headerDividersEnabled

    创建ListView

    (一)、ArrayAdapter实现单行文本ListView:

    1、使用步骤。

    • 定义一个数组来存放ListView中item的内容;
    • 通过实现ArrayAdapter的构造方法创建一个ArrayAdapter对象;
    • 通过ListView的setAdapter()方法绑定ArrayAdapter。

    ArrayAdapter有多个构造方法,最常用三个参数的那种。

    • 第一个参数:上下文对象;
    • 第二个参数:ListView的每一行(也就是item)的布局资源id;
    • 第三个参数:ListView的数据源。

    2、使用系统自带布局文件的不同效果:

    • A、android.R.layout.simple_list_item_1
    • B、android.R.layout.simple_list_item_checked
    • C、android.R.layout.simple_list_item_multiple_choice
    • D、android.R.layout.simple_list_item_single_choice

    3、核心代码:

    String[] strArr = new String[] { "yuhongxing", "sunshengling",
                                "chenyanzhang", "huangchao", "liupengfei" };
    listView_main_userList = (ListView) findViewById(R.id.listView_main_userlist);
    ArrayAdapter<String> adapter = new ArrayAdapter<String>(
                MainActivity.this, android.R.layout.simple_list_item_1, strArr);
    listView_main_userList.setAdapter(adapter);
    

    ListView的监听器与Spinner的监听器的区别:

    • Spinner是:setOnItemSelectedListener
    • ListView是:setOnItemClickListener

    这两个监听器是否可以互换使用呢?

    • 在Spinner中使用OnItemClickListener会异常。java.lang.RuntimeException: setOnItemClickListener cannot be used with a spinner。而如果在ListView中使用OnItemSelectedListener,则没有反应,也就是说该监听器不会被触发执行;
    • OnItemSelectedListener 监听器的回调方法中,parent.getSelectedItem()和parent.getItemAtPosition(position)都能返回object对象。而OnItemClickListener监听器的回调方法中parent.getSelectedItem()只能返回null。

    (二)、 SimpleAdapter 实现多行文本ListView:

    1、使用步骤。

    • 定义一个集合来存放ListView中item的内容;
    • 定义一个item的布局文件;
    • 创建一个 SimpleAdapter 对象;
    • 通过ListView的setAdapter()方法绑定 SimpleAdapter  。

    2、核心代码:

    public class MainActivity extends Activity {
    	private static final String TAG = "MainActivity";
    	private ListView listView_main_news;
    	private List<Map<String, String>> list = null;
       	@Override
    	protected void onCreate(Bundle savedInstanceState) {
           super.onCreate(savedInstanceState);
           setContentView(R.layout.activity_main);
           listView_main_news = (ListView) findViewById(R.id.listView_main_news);
           list = new ArrayList<Map<String, String>>();
           for (int i = 0; i < 5; i++) {
                   Map<String, String> map = new HashMap<String, String>();
                  map.put("username", "wangxiangjun_" + i);
                  map.put("password", "123456_" + i);
                   list.add(map);
           }
               Log.i(TAG, "==" + list.toString());
          // 定义SimpleAdapter适配器。
          // 使用SimpleAdapter来作为ListView的适配器,比ArrayAdapter能展现更复杂的布局效果。为了显示较为复杂的ListView的item效果,需要写一个xml布局文件,来设置ListView中每一个item的格式。
               SimpleAdapter adapter = new SimpleAdapter(this, list,
                           R.layout.item_listview_main, new String[] { "username",
                                        "password" }, new int[] {
                                           R.id.text_item_listview_username,
                                           R.id.text_item_listview_pwd });
           listView_main_news.setAdapter(adapter);
    }
    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
           getMenuInflater().inflate(R.menu.main, menu);
                return true;
        }
    }
    

    (三)、SimpleAdapter实现多行文本且带图片ListView:

    1、使用步骤。

    • 定义一个集合来存放ListView中item的内容;
    • 定义一个item的布局文件;
    • 创建一个 SimpleAdapter 对象;
    • 通过ListView的setAdapter()方法绑定 SimpleAdapter  。

    2、核心代码:

    public class MainActivity extends Activity {
    private ListView listView_main_regmsg;
    private int[] imgIds = new int[] { R.drawable.pic01, R.drawable.pic02,
                   R.drawable.pic03, R.drawable.pic04 };
       @Override
    protected void onCreate(Bundle savedInstanceState) {
          super.onCreate(savedInstanceState);
           setContentView(R.layout.activity_main);
          listView_main_regmsg = (ListView) findViewById(R.id.listView_main_regmsg);
                // 创建数据源
           List<Map<String, Object>> list = new ArrayList<Map<String, Object>>();
           for (int i = 0; i < imgIds.length; i++) {
                   Map<String, Object> map = new HashMap<String, Object>();
                   map.put("username", "wanglu_" + i);
                   map.put("pwd", "123456_" + i);
                   map.put("imgId", imgIds[i]);
                   list.add(map);
           }
           /** 常用的SimpleAdapter的构造方法有五个参数:
          * @param context :表示上下文对象或者环境对象。
          * @param data :表示数据源。往往采用List<Map<String, Object>>集合对象。
          * @param resource :自定义的ListView中每个item的布局文件。用R.layout.文件名的形式来调用。
          * @param from :其实是数据源中Map的key组成的一个String数组。
          * @param to :表示数据源中Map的value要放置在item中的哪个控件位置上。其实就是自定义的item布局文件中每个控件的id。
          * 通过R.id.id名字的形式来调用。
          */
           SimpleAdapter adapter = new SimpleAdapter(this, list,
                           R.layout.item_listview_main, new String[] { "username", "pwd",
                                           "imgId" }, new int[] {
                                           R.id.text_item_listview_username,
                                           R.id.text_item_listview_pwd,
                                           R.id.imageView_item_listview_headpic });
                // 给ListView设置适配器
           listView_main_regmsg.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.main, menu);
                return true;
    	}
    }
    

    BaseAdapter自定义适配器实现ListView:

    1、使用步骤

    • 定义一个集合来存放ListView中item的内容;

    • 定义一个item的布局文件;

    • 定义一个 继承了BaseAdapter的子类MyAdapter,重写未实现的方法;(定义ViewHolder,重写getView()方法)

    • 创建一个内部类:MyAdapter extends BaseAdapter;

        - 实现未实现的方法:getCount() 、getItem()、 getItemId()、 getView();
        - 定义内部类ViewHolder,将item布局文件中的控件都定义成属性;
        - 构建一个布局填充器对象:LayoutInflater.from(context);
        - 调用布局填充器对象的inflate()方法填充item布局文件,将返回的view对象赋值给convertView;
        - 调用convertView对象的findViewById()获取item布局中的控件,将控件对象赋值给ViewHolder中的属性;
        - 给convertView对象设置标签,也就是调用setTag()方法,将ViewHolder对象作为标签贴在convertView对象上;
        - 从根据convertView的标签,从convertView对象上取回ViewHolder对象。
      
    • 通过ListView的setAdapter()方法绑定自定义的MyAdapter对象 。

    2、核心代码:

    class MyAdapter extends BaseAdapter {
        private Context context = null;
        public MyAdapter(Context context) {
                this.context = context;
        }
        @Override
        public int getCount() {
                return list.size();
        }
        @Override
        public Object getItem(int position) {
                return list.get(position);
        }
        @Override
        public long getItemId(int position) {
                return position;
        }
        @Override
        public View getView(int position, View convertView, ViewGroup parent) {
                ViewHolder mHolder;
                if (convertView == null) {
                        mHolder = new ViewHolder();
                        LayoutInflater inflater = LayoutInflater.from(context);
                        convertView = inflater.inflate(R.layout.item_listview_main_userlist, null, true);
                        mHolder.text_item_listview_username = (TextView) convertView.findViewById(R.id.text_item_listview_username);
                        mHolder.text_item_listview_email = (TextView) convertView.findViewById(R.id.text_item_listview_email);
                        mHolder.imageView_item_listview_headpic = (ImageView) convertView.findViewById(R.id.imageView_item_listview_headpic);
                        convertView.setTag(mHolder);
                } else {
                        mHolder = (ViewHolder) convertView.getTag();
                }
                String username = list.get(position).get("username").toString();
                String email = list.get(position).get("email").toString();
                int picId = Integer.parseInt(list.get(position).get("headpic").toString());
          mHolder.text_item_listview_username.setText(username);
                mHolder.text_item_listview_email.setText(email);
                mHolder.imageView_item_listview_headpic.setImageResource(picId);
                return convertView;
        }
    
        class ViewHolder {
                private TextView text_item_listview_username;
                private TextView text_item_listview_email;
                private ImageView imageView_item_listview_headpic;
        }
    }
    

    (五)、convertView原理:

    • Adapter的作用就是ListView界面与数据之间的桥梁,当列表里的每一项显示到页面时,都会调用Adapter的getView方法返回一个View。

    • 如果在我们的列表有上千项时会是什么样的?是不是会占用极大的系统资源?

    • Android中有个叫做Recycler的构件,下图是他的工作原理:

        - 如果你有100个item,其中只有可见的项目存在内存中,其他的在Recycler中。
        - ListView先请求一个type1视图(getView),然后请求其他可见的item,convertView在getView中是空(null)的。
        - 当item1滚出屏幕,并且一个新的item从屏幕底端上来时,ListView再请求一个type1视图,convertView此时不是空值了,它的值是item1。你只需设定新的数据,然后返回convertView,不必重新创建一个视图。
      

  • 相关阅读:
    VUE单页模板
    Nacos(作为配置中心)
    OpenFeign与Nacos(作为注册中心) 远程调用
    Nacos 服务注册
    Docker 安装Redis
    Docker安装Mysql
    多线程之CAS和ABA
    Volatile
    JAVA语言的特点
    判断多选按钮被选中两种方法
  • 原文地址:https://www.cnblogs.com/s-y-j/p/6548032.html
Copyright © 2011-2022 走看看