目录

一、ListView的简单用法

二、进一步优化ListView界面

三、提升ListView的效率(优化)

四、ListView的点击事件


本篇文章主要讲解最常用的控件之一:ListView的使用。

        当我们的程序有大量的数据需要展示的时候,可以借助ListView实现。ListView允许用户上下滑动滚动屏幕,如:查看QQ聊天记录。

一、ListView的简单用法

 首先新建一个新的活动,7.xml中的代码如下:

        布局中引入ListView控件和前面其他控件用法一致。

<?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">

    <ListView
        android:id="@+id/list_view"
        android:layout_width="match_parent"
        android:layout_height="match_parent"/>

</LinearLayout>

7.java代码如下:

        首先准备好一个数据数组用来展示,通过适配器将数据传给ListView,ArrayAdapter的构造函数中依次传入上下文、ListView子项布局的id,要适配的数据。注意:这里使用的android.R.layout.simple_list_item_1为Android内置的布局文件,里面只有一个TextView,用于简单显示一段文本。最后调用ListView的setAdapter()方法,将适配器对象传进去,这样ListView和数据之间的关联就建立完成。

public class 7 extends AppCompatActivity {

    private String[] data = {"A","B","C","D","E","F","G","H","I","J","K","L","M","N"};
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_7);
        ArrayAdapter<String> adapter = new ArrayAdapter<String>(
                7.this, android.R.layout.simple_list_item_1, data);
        ListView listView = findViewById(R.id.list_view);
        listView.setAdapter(adapter);
    }
}

效果如下:

android listview设置item的点击效果 安卓listview用法_android

二、进一步优化ListView界面

        只显示一段文本的ListView很单调,下面对ListView界面添加图片进行美观。效果如下:

android listview设置item的点击效果 安卓listview用法_android studio_02

 

新建一个Image类,作为ListView适配器的适配类型,代码如下:

        name表示每行字母的名字,imageId表示图片id。

public class Image {
    private String name;
    private int imageId;
    
    public Image(String name,int imageId){
        this.name = name;
        this.imageId = imageId;
    }
    public String getName(){
        return name;
    }
    public int getImageId(){
        return imageId;
    }
}

 接下来需要为ListView子项指定自定义的布局,在layout目录下新建item.xml,代码如下:

<?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="wrap_content">
    
    <ImageView
        android:id="@+id/image"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"/>
    
    <TextView
        android:id="@+id/name"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center_vertical"
        android:layout_marginLeft="10dp"/>

</LinearLayout>

接下来创建一个自定义的适配器adapter类,,代码如下:

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

        重写getView()方法,此方法在每个子项被滚动到屏幕内时调用。

        在getView()方法中,首先通过getItem方法得到当前项的image实例,然后使用LayoutInflater方法来为这个子项加载传入的布局,LayoutInflater方法的inflate方法接收3个参数,第三个参数指定为false,表示只让在父布局中声明的layout属性生效,接下来获取到两个实例,并分别调用setImageResource和setText方法设置显示的图片和文字,最后将布局返回,这样自定义的适配器就完成了。

public class adapter extends ArrayAdapter<Image> {
    private int resourceId;
    public adapter(Context context, int textViewResourceId, List<Image> objects){
        super(context,textViewResourceId,objects);
        resourceId = textViewResourceId;
    }
    
    @Override
    public View getView(int position, @Nullable View convertView, @NonNull ViewGroup parent) {
        Image image = getItem(position);
        View view = LayoutInflater.from(getContext()).inflate(resourceId,parent,false);
        ImageView image2 = view.findViewById(R.id.image);
        TextView name = view.findViewById(R.id.name);
        image2.setImageResource(image.getImageId());
        name.setText(image.getImageId());
        return view;
    }
}

下面修改主类7.java的代码如下:

        首先,自定义一个init()方法初始化list中的元素,然后创建自定义适配器adapter的实例,最后将适配器传递给ListView,这样就完成了。

public class 7 extends AppCompatActivity {

    private List<Image> list = new ArrayList<>();
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_7);
        init();//初始化数据
        adapter ad = new adapter(7.this,R.layout.item,list);
        ListView listview = findViewById(R.id.list_view);
        listview.setAdapter(ad);
    }
    
    private void init(){
        for(int i = 0; i < 5; i++){
            Image a = new Image("A",R.drawable.image1);
            list.add(a);
            Image b = new Image("B",R.drawable.image1);
            list.add(b);
            Image c = new Image("C",R.drawable.image1);
            list.add(c);
            Image d = new Image("D",R.drawable.image1);
            list.add(d);
            Image e = new Image("E",R.drawable.image1);
            list.add(e);
        }
    }
}

        现在只用修改item.xml文件中的内容,就可以做出各种复杂的界面了。

三、提升ListView的效率(优化)

        目前,上面的代码还有待优化的地方,如:adapter的getView()方法中,每次都将布局重新加载一遍,当ListView快速滚动的时候,运行效率会降低。再比如:每次在getView()方法中都会调用findViewById()方法获取控件实例。下面就对这两个问题进行优化。

修改adapter中的getView()方法中的代码,如下:

        getView()方法有一个convertView参数,这个参数用于将之前加载好的布局进行缓存,如果为null则加载布局,如果不为null则直接重用convertView。这样优化了第一个问题。

public View getView(int position, @Nullable View convertView, @NonNull ViewGroup parent) {
        Image image = getItem(position);
        View view;
        if(convertView == null){
            view = LayoutInflater.from(getContext()).inflate(resourceId,parent,false);
        }else{
            view = convertView;
        }
        ImageView image2 = view.findViewById(R.id.image);
        TextView name = view.findViewById(R.id.name);
        image2.setImageResource(image.getImageId());
        name.setText(image.getName());
        return view;
    }

再次修改getView()方法中的代码,如下:

        新增一个内部类ViewHolder,用于对控件的实例进行缓存,当convertView为null时,创建一个ViewHolder对象,将控件的实例都存放在ViewHolder里,调用setTag()方法,将ViewHolder对象存放在View中;不为null时,调用getTag,重新取出。这样优化了第二个问题。

public View getView(int position, @Nullable View convertView, @NonNull ViewGroup parent) {
        Image image = getItem(position);
        View view;
        ViewHolder viewHolder;
        if(convertView == null){
            view = LayoutInflater.from(getContext()).inflate(resourceId,parent,false);
            viewHolder = new ViewHolder();
            viewHolder.image = view.findViewById(R.id.image);
            viewHolder.name = view.findViewById(R.id.name);
            view .setTag(viewHolder);

        }else{
            view = convertView;
            viewHolder = (ViewHolder) view.getTag();
        }
        viewHolder.image.setImageResource(image.getImageId());
        viewHolder.name.setText(image.getName());
        return view;
    }
    class ViewHolder{
        ImageView image;
        TextView name;
    }

四、ListView的点击事件

        点击事件是不可或缺的,前面的只是满足视觉上的效果,点击事件才是实际用途。

修改7.java的代码,如下:

        原理同前面按钮的点击事件一致,创建实例后,调用setOnItemClickListener()方法为ListView注册监听器,通过参数i判断用户点击的子项。

protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_7);
        init();//初始化数据
        adapter ad = new adapter(7.this,R.layout.item,list);
        ListView listview = findViewById(R.id.list_view);
        listview.setAdapter(ad);
        listview.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> adapterView, View view, int i, long l) {
                Image image = list.get(i);
                Toast.makeText(7.this, image.getName(), Toast.LENGTH_SHORT).show();
            }
        });
    }

效果如下:

android listview设置item的点击效果 安卓listview用法_Image_03

 


希望本文章对你有帮助,如果你对Android开发感兴趣,请持续关注本专栏,帮助你从入门到项目实战,你将收获:Android基础开发、各种经典功能实现、项目实战、开发自己的APP、将APP上传应用商店、靠广告赚钱等等,持续更新ing......