ListView

•前言

  ListView 绝对可以称得上是 Android 中最常用的控件之一,几乎所有的应用程序都会用到它。

  由于手机屏幕空间有限,能够一次性在屏幕上显示的内容并不多,当我们的程序中有大量的数据需要展示的时候,就可以借助 ListView 来实现。

  ListView 允许用户通过手指上下滑动的方式将屏幕外的数据滚动到屏幕内,同时屏幕上原有的数据则会滚动出屏幕。

  其实你每天都在使用这个控件,比如查看 QQ聊天记录,翻阅微博消息,等等。

•ListView简介

  ListView 的直接父类是 View.Group,也就是说,他自己定义了子排列 View 的规则。

  ListView 和所要展示的内容(数据源)之间需要 Adapter(适配器) 来实现。

  Adapter 是一个桥梁,对 ListView 的数据进行管理。

  数据来源不同,所使用的 Adapter 也不同,数据源(Data source)、Adapter和列表(ListView)之间的关系如下图所示:

    
    

•ListView相关属性

  • android:dividerHeight="2dp" : 设置分割线高度
  • android:divider="@color/red" : 设置分割条,可以用颜色分割,也可以用 drawable 资源分割
  • android:entries="@array/myarray" : 设置 ListView 显示的内容

•ListView的简单用法

  在 res/values 下创建一个 arrays.xml 文件,添加代码如下:

<?xml version="1.0" encoding="utf-8"?>
<resources>
<string-array name="myarray">
<item>关羽</item>
<item>孙尚香</item>
<item>娜可露露</item>
</string-array>
</resources>

  新建一个 Activity,添加代码如下:

<?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"
android:orientation="vertical"
android:padding="10dp"> <TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Test Array Adapter"
android:textSize="20sp"
/> <!-- 为 ListView 设置红色的分割线
并将分割线宽度设置为 2dp -->
<ListView
android:id="@+id/lv_array_adapter"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="10dp"
android:divider="@color/red"
android:dividerHeight="2dp"
android:entries="@array/myarray"
/> </LinearLayout>

•运行效果

  

•Adapter简介

  Adapter 的继承关系如下图所示:

    

  Adapter 是一个接口,ListAdapter 继承了 Adapter,也是一个接口,并需要子类实现。

  BaseAdapter 实现了 ListAdapter,他是一个抽象类。

  SimpleAdapter 继承自 BaseAdapter,他是 Adapter 的一个实例对象。

  另外,还有 ArrayAdapter 和 SimpleCursorAdapter 也是 Adapter 的实例对象。

    • ArrayAdapter:支持泛型操作,最简单的一个Adapter,只能展现一行文字
    • SimpleAdapter:同样具有良好扩展性的一个Adapter,可以自定义多种效果
    • BaseAdapter:抽象类,实际开发中我们会继承这个类并且重写相关方法,用得最多的一个Adapter
    • SimpleCursorAdapter:用于显示简单文本类型的listView,一般在数据库那里会用到,不过有点过时, 不推荐使用

ArrayAdapter

•示例一

  在 res/layout 新建 activity_array_adapter.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"
android:orientation="vertical"
android:padding="10dp"> <TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Test Array Adapter"
android:textSize="20sp"
/> <!-- 为 ListView 设置红色的分割线
并将分割线宽度设置为 2dp -->
<ListView
android:id="@+id/lv_array_adapter"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="10dp"
android:divider="@color/red"
android:dividerHeight="2dp"
/> </LinearLayout>

  新建 ArrayAdapterActivity.java 文件,添加代码如下:

public class ArrayAdapterActivity extends AppCompatActivity {

    private ListView listview;

    @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_array_adapter); String[] s = new String[]{"关羽", "孙尚香", "娜可露露"};
ArrayAdapter<String> adapter = new ArrayAdapter<>(this, R.layout.array_adapter_item, s); listview = findViewById(R.id.lv_array_adapter);
listview.setAdapter(adapter);
} }

  在这段代码中,定义了一个字符串数组 s ,不过数组 s 中的数据是无法直接传递给 ListView 的;

  我们还需要借助适配器来完成(这里我们借助 ArrayAdapter 这个适配器);

  ArrayAdapter 可以通过泛型来指定要适配的数据类型,然后再构造函数中把要适配的数据传入;

  ArrayAdapter 有多个构造函数的重载,我们应该根据实际情况选择最合适的一种;

  这里由于我们提供的数据都是字符串,因此将 ArrayAdapter 的泛型指定为 String;

  然后在 ArrayAdapter 的构造函数中依次传入:

    • 当前上下文(this)
    • ListView子项布局的 id(R.layout.array_adapter_item)
    • 适配的数据(s)

  R.layout.array_adapter_item 布局代码如下:

<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:id="@+id/textView"
android:textSize="20sp"
android:textColor="@color/black"> </TextView>

  此布局用来设置显示的字体(关羽、孙尚香、娜可露露)风格。

•运行效果

  

•示例二

  只能显示一段文本的 ListView 实在是太单调了,我们现在就来对 ListView 的界面进行定制,让它可以显示更加丰富的内容。

  首先需要准备一组图片,分别对应上面提供的英雄:

              

          $guan\_yu.jpg$     $sun\_shang\_xiang.jpg$  $na\_ke\_lu\_lu.jpg$

  接着定义一个实体类,作为 ListView 适配器的适配类型。

  新建类 Person,代码如下:

public class Person {
private String name;//英雄名称
private int imgId;//对应图片id public Person(String name,int imgId){
this.name = name;
this.imgId = imgId;
} public String getName() {
return name;
} public int getImgId() {
return imgId;
}
}

  Person 类中只有两个字段,name 表示英雄名称,imgId 表示英雄对应图片的资源 id。

  然后需要为 ListView 的子项指定一个我们自定义的布局;

  在 layout 目录下新建 person_item,添加代码如下:

<?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"
android:padding="10dp"
android:layout_marginTop="20dp"> <ImageView
android:id="@+id/person_img"
android:layout_width="100dp"
android:layout_height="150dp"
android:scaleType="centerCrop"/> <TextView
android:id="@+id/person_name"
android:layout_width="match_parent"
android:layout_height="150dp"
android:layout_marginLeft="10dp"
android:gravity="center"
android:textSize="20sp"
android:textColor="@color/red"
/> </LinearLayout>

  在这个布局中,我们定义了一个 ImageView 用来显示图片,有定义了一个 TextView 用来显示名称。

  接下来需要创建一个自定义的适配器,这个适配器继承自 ArrayAdapter,并将泛型指定为 Person 类。

  新建 PersonAdapter 类,添加代码如下:

public class PersonAdapter extends ArrayAdapter<Person> {

    private Context context;
private int resource; public PersonAdapter(@NonNull Context context, int resource, @NonNull List<Person> objects) {
super(context, resource, objects);
this.context = context;
this.resource = resource;
} @NonNull
@Override
public View getView(int position, @Nullable View convertView, @NonNull ViewGroup parent) {
Person person = getItem(position);//获取当前项的 Person 实例 View view = LayoutInflater.from(context).inflate(resource, parent, false);
ImageView img = view.findViewById(R.id.person_img);
TextView name = view.findViewById(R.id.person_name); img.setImageResource(person.getImgId());
name.setText(person.getName()); return view;
}
}

  PersonAdapter 重写了父类的一组构造函数,用于将上下文、ListView 子项布局 id 和数据都传递进来。

  另外又重写了 getView() 方法,这个方法在每个子项被滚动到屏幕内的时候被调用。

  在 getView() 方法中,首先通过 getItem() 方法得到当前项的 Person 实例,然后使用 LayoutInflater 来为这个子项加载我们传入的布局。

  通过 LayoutInflater 的 from() 方法可以构建出一个 LayoutInflater 对象,然后调用 inflate() 方法动态加载一个布局文件。

  inflate() 方法接收三个参数:

    • 第一个参数是要加载的布局文件的 id(resource)
    • 第二个参数是给加载好的布局再添加一个父布局(parent)
    • 第三个参数指定成 false

  接下来调用 view 的 findViewByid() 方法分别获取到 ImageView 和 TextView 的实例。

  并分别调用他们的 setImageResource() 和 setText() 方法来设置现实的图片和文字。

  最后将布局返回,这样我们的适配器就完成了。

  最后修改 ArrayAdapterActivity.java 中的代码,如下所示:

public class ArrayAdapterActivity extends AppCompatActivity {

    private ListView listview;

    @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_array_adapter); PersonAdapter adapter = new PersonAdapter(ArrayAdapterActivity.this, R.layout.person_item, getData()); listview = findViewById(R.id.lv_array_adapter);
listview.setAdapter(adapter);
} private List<Person> getData() {
List<Person> list = new ArrayList<>(); Person guanYu = new Person("关羽", R.drawable.guan_yu);
list.add(guanYu); Person sunShangXiang = new Person("孙尚香", R.drawable.sun_shang_xiang);
list.add(sunShangXiang); Person naKeLL = new Person("娜可露露", R.drawable.na_ke_lu_lu);
list.add(naKeLL); return list;
}
}

  可以看到,这里添加了一个 getData() 方法,用于获取数据。

  接着在 onCreate() 方法中创建了 PersonAdapter 对象,并将 PersonAdapter 作为适配器传递个 ListView。

  这样定值 ListView 界面的任务就完成了。

•运行效果

  

•提升ListView 的运行效率

  之所以说 ListView 这个控件很难用,是因为它有很多细节可以优化,其中运行效率就是很重要的一点;

  目前我们的 ListView 运行效率是很低的,因为在 PersonAdapter 的 getView() 方法中,每次都将布局重新加载了一遍;

  当 ListView 快速滚动的时候,这就会成为性能的瓶颈;

  仔细观察你会发现,getView() 方法中有一个 convertView 参数;

  这个参数用于将之前加载好的布局进行缓存,以便之后可以重用。

  修改 PersonAdapter 中的 getView() 代码,如下所示:

public class PersonAdapter extends ArrayAdapter<Person> {

    ......

    @NonNull
@Override
public View getView(int position, @Nullable View convertView, @NonNull ViewGroup parent) {
Person person = getItem(position);//获取当前项的 Person 实例
View view;
if (convertView == null) {
view = LayoutInflater.from(context).inflate(resource, parent, false);
} else {
view = convertView;
} ImageView img = view.findViewById(R.id.person_img);
TextView name = view.findViewById(R.id.person_name); img.setImageResource(person.getImgId());
name.setText(person.getName()); return view;
}
}

  可以看到,现在我们在 getView() 方法中进行了判断,如果 convertView 为 null,则使用 LayoutInflater 去加载布局;

  如果不为空,这直接对 convertView 进行重用;

  这样就大大提高了 ListView 的运行效率,在快速滚动的时候也可以表现出更好的性能。

  不过,目前我们的这份代码还是可以继续优化的;

  虽然现在已经不会再重复去加载布局,但是每次在 getView() 方法中还是会调用 View 的 findViewById() 方法来获取一次控件的实例;

  我们可以借助 ViewHolder 来对这部分性能进行优化;

  修改 PersonAdapter 中的 getView() 代码,如下所示:

public class PersonAdapter extends ArrayAdapter<Person> {

    private Context context;
private int resource; public PersonAdapter(@NonNull Context context, int resource, @NonNull List<Person> objects) {
super(context, resource, objects);
this.context = context;
this.resource = resource;
} static class ViewHolder {
ImageView img;
TextView name;
} @NonNull
@Override
public View getView(int position, @Nullable View convertView, @NonNull ViewGroup parent) {
Person person = getItem(position);//获取当前项的 Person 实例
View view;
ViewHolder viewHolder;
if (convertView == null) {
view = LayoutInflater.from(context).inflate(resource, parent, false);
viewHolder = new ViewHolder();
viewHolder.img = view.findViewById(R.id.person_img);
viewHolder.name = view.findViewById(R.id.person_name);
view.setTag(viewHolder);//将 viewHolder 存储在 View 中
} else {
view = convertView;
viewHolder = (ViewHolder) view.getTag();
}
viewHolder.img.setImageResource(person.getImgId());
viewHolder.name.setText(person.getName()); return view;
}
}

  我们新增了一个静态内部类 ViewHolder,用于对控件的实例进行缓存。

  当 convertView 为 null 的时候,创建一个 ViewHolder 对象,并将控件的实例都存放在 viewHolder 里;

  然后调用 view 的 setTag() 方法,将 viewHolder 对象存储在 view 中;

  当 convertView 不为 null 时,则调用 view.getTag() 方法,把 viewHolder 重新取出;

  这样所有的控件的实例都缓存在了 viewHolder 里,就没有必要每次都通过 findViewById() 方法来获取控件实例了。

  另外这个修饰 ViewHolder 的 static,关于是否定义成静态,跟里面的对象数目是没有关系的;

  加静态是为了在多个地方使用这个 viewHolder 的时候,类只需加载一次,如果只是使用了一次,加不加也无所谓!

                                          ——Berial(B神)原话~

•为 ListView 设置点击事件

  修改 ArrayAdapterActivity.java 中的代码,如下所示:

public class ArrayAdapterActivity extends AppCompatActivity {

    private List<Person> personList;
private ListView listview; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_array_adapter); personList = getData();
PersonAdapter adapter = new PersonAdapter(ArrayAdapterActivity.this, R.layout.person_item, personList); listview = findViewById(R.id.lv_array_adapter);
listview.setAdapter(adapter); listview.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
Person person = personList.get(position);
Toast.makeText(ArrayAdapterActivity.this,person.getName()+"被点击了!",Toast.LENGTH_SHORT).show();
}
});
} private List<Person> getData() {
List<Person> list = new ArrayList<>(); Person guanYu = new Person("关羽", R.drawable.guan_yu);
list.add(guanYu); Person sunShangXiang = new Person("孙尚香", R.drawable.sun_shang_xiang);
list.add(sunShangXiang); Person naKeLL = new Person("娜可露露", R.drawable.na_ke_lu_lu);
list.add(naKeLL); return list;
}
}

  可以看到,我们使用 setOnItemClickListener() 方法为 ListView 注册了一个监听器;

  当用户点击了 ListView 中的任何一个子项时,就会调用 onItemClick() 方法;

  在这个方法中可以通过 position 参数判断出用户点击的是哪一个子项,然后获取到相应的 Person 实例;

  最后通过 Toast 将其显示出来;

•运行效果

  

最新文章

  1. (String)151. Reverse Words in a String
  2. Linux MD5值递归比对目录中的文件是否有修改
  3. (转)supertable像excel那样固定table的表头和第一列
  4. 鸟哥的linux私房菜之vim
  5. Webform——服务器控件与客户端控件
  6. npm 模块安装机制简介
  7. 21.编写一个Java应用程序,该程序包括3个类:Monkey类、People类和主类 E。要求: (1) Monkey类中有个构造方法:Monkey (String s),并且有个public void speak() 方法,在speak方法中输出“咿咿呀呀......”的信息。 (2)People类是Monkey类的子类,在People类中重写方法speak(),在speak方法 中输出“小样
  8. [bzoj1227] [SDOI2009]虔诚的墓主人
  9. LNMP Yii2 验证码不显示问题最终解决方案
  10. Jmeter入门(01)Jmeter的下载和安装
  11. DevExpress v18.1新版亮点——DevExtreme篇(四)
  12. JavaSE——多线程
  13. out参数ref参数params 可变参数
  14. Node.js从入门到实战ECMAScript6一页纸总结(很大的一页纸)
  15. wikioi 1017 乘积最大
  16. Vert.x中EventBus中的使用
  17. es6新语法Object.assign()
  18. Tengine是由淘宝网发起的Web服务器项目。它在Nginx的基础上,针对大访问量网站的需求,添加了很多高级功能和特性
  19. hash值是啥,单页面应用的实质
  20. C#代码实现 Excel表格与Object互相转换,Excel表格导入数据库(.NET2.0 .NET4.0)

热门文章

  1. COOP &amp; COEP
  2. tslint 忽略对某行代码的检测
  3. NGK公链生态所如何保障用户的数字资产隐私安全?
  4. Nice!JavaScript基础语法知识都在这儿了
  5. iOS拍照之系统拍照
  6. idea加载maven项目遇见的坑---2
  7. Oracle数据库的函数
  8. Kubernetes中分布式存储Rook-Ceph部署快速演练
  9. How DRI and DRM Work
  10. 后端程序员之路 23、一个c++的api framework