ListView第三节,使用自定义适配器实现图文列表以及使用 convertView和viewHolder进行优化

除了使用系统原有的适配器,我们也可以自定义适配器来实现我们想要的效果

其他步骤都一样,只不过需要重写一个继承BaseAdapter的内部类

package com.example.listviewtest02;

import androidx.appcompat.app.AppCompatActivity;

import android.content.Context;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.TextView;

public class MainActivity extends AppCompatActivity {
    private ListView listView;
    //还是首先声明listview,然后在oncreate里面绑定资源文件
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        listView = findViewById(R.id.listView);
         //然后在oncreate里面绑定资源文件
        listView.setAdapter(new myAdapter(this));
         //最后把做好的adapter放进listview即可
       
    }

    static class myAdapter extends BaseAdapter{
        private String[] titles = {"客户-1","客户-2","客户-3","客户-4","客户-5","客户-6","客户-7"};
        private int[] icons = {
            android.R.drawable.ic_input_add,
            android.R.drawable.ic_delete,
            android.R.drawable.ic_dialog_email,
            android.R.drawable.ic_input_delete,
            android.R.drawable.ic_input_get,
            android.R.drawable.ic_dialog_map,
            android.R.drawable.ic_dialog_info

        };

        private Context context;

        public myAdapter(Context context){
            this.context = context;
        }
        @Override
        public int getCount() {
            return titles.length;
        }

        @Override
        public Object getItem(int i) {
            return titles[i];
        }

        @Override
        public long getItemId(int i) {
            return i;
        }

        @Override
        public View getView(int i, View view, ViewGroup viewGroup) {
            LayoutInflater inflater = LayoutInflater.from(context);
             view = inflater.inflate(R.layout.item,null);
            TextView textView = view.findViewById(R.id.textView2);
            ImageView imageView = view.findViewById(R.id.imageView);

            textView.setText(titles[i]);
            imageView.setImageResource(icons[i]);

            return view;

        }
    }

}

因为listview每一次在屏幕出现新的项目都会在内存新建新的对象,如果进行大量的屏幕滑动就会产生大量的垃圾对象从而降低性能,可以使用convertView进行优化,因为可以同屏出现条目时固定的,当新的条目出现时旧的就会消失,我们可以对消失的对象进行复用,这样只需要固定数量的对象即可满足所有条目。

只需要在getview里面加一个判断,如果view为空的话再新建view,如果有现成的就用现成的

package com.example.listviewtest02;

import androidx.appcompat.app.AppCompatActivity;

import android.content.Context;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.TextView;

public class MainActivity extends AppCompatActivity {
    private ListView listView;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        listView = findViewById(R.id.listView);
        listView.setAdapter(new myAdapter(this));
    }

    static class myAdapter extends BaseAdapter{
        private String[] titles = {"客户-1","客户-2","客户-3","客户-4","客户-5","客户-6","客户-7"};
        private int[] icons = {
            android.R.drawable.ic_input_add,
            android.R.drawable.ic_delete,
            android.R.drawable.ic_dialog_email,
            android.R.drawable.ic_input_delete,
            android.R.drawable.ic_input_get,
            android.R.drawable.ic_dialog_map,
            android.R.drawable.ic_dialog_info

        };

        private Context context;

        public myAdapter(Context context){
            this.context = context;
        }
        @Override
        public int getCount() {
            return titles.length;
        }

        @Override
        public Object getItem(int i) {
            return titles[i];
        }

        @Override
        public long getItemId(int i) {
            return i;
        }

        @Override
        public View getView(int i, View view, ViewGroup viewGroup) {

            if (view == null){

                LayoutInflater inflater = LayoutInflater.from(context);
                view = inflater.inflate(R.layout.item,null);
            }
            TextView textView = view.findViewById(R.id.textView2);
            ImageView imageView = view.findViewById(R.id.imageView);

            textView.setText(titles[i]);
            imageView.setImageResource(icons[i]);

            return view;

        }
    }

}

做完这一步后我们还会发现,虽然每一次新出现的组件不用每次都新建了,但是在给旧对象重新赋值的过程中我们还会一直重复的在系统中查找组件来赋给对象,这一部分也可以优化,就是自己建一个类把组件放进去,这样我们只需要程序启动的时候查找一次然后保存在内部类里面,以后再用的话可以直接用保存下来的值直接赋给对象就行了。

package com.example.listviewtest02;

import androidx.appcompat.app.AppCompatActivity;

import android.content.Context;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.TextView;

public class MainActivity extends AppCompatActivity {
    private ListView listView;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        listView = findViewById(R.id.listView);
        listView.setAdapter(new myAdapter(this));
    }

    static class myAdapter extends BaseAdapter{


        private String[] titles = {"客户-1","客户-2","客户-3","客户-4","客户-5","客户-6","客户-7"};
        private int[] icons = {
            android.R.drawable.ic_input_add,
            android.R.drawable.ic_delete,
            android.R.drawable.ic_dialog_email,
            android.R.drawable.ic_input_delete,
            android.R.drawable.ic_input_get,
            android.R.drawable.ic_dialog_map,
            android.R.drawable.ic_dialog_info

        };

        private Context context;

        public myAdapter(Context context){
            this.context = context;
        }
        @Override
        public int getCount() {
            return titles.length;
        }

        @Override
        public Object getItem(int i) {
            return titles[i];
        }

        @Override
        public long getItemId(int i) {
            return i;
        }

        @Override
        public View getView(int i, View view, ViewGroup viewGroup) {
            //先声明一个内部类
            ViewHolder viewHolder;
            if (view == null){

                LayoutInflater inflater = LayoutInflater.from(context);
                view = inflater.inflate(R.layout.item,null);
                //然后实例化
                viewHolder = new ViewHolder();
                //再把图片和文字的组件地址赋值
                viewHolder.iv_icon = view.findViewById(R.id.imageView);
                viewHolder.tv_title = view.findViewById(R.id.textView2);    
                //最后把赋好值的viewholder赋值给view
                view.setTag(viewHolder);
            }else {
                //上面是如果没有赋过值就复制,这里如果已经赋值了就直接把赋过得值拿来用就行
                viewHolder = (ViewHolder) view.getTag();
            }

            //然后再把数组里的内容填进组件
            viewHolder.tv_title.setText(titles[i]);
            viewHolder.iv_icon.setImageResource(icons[i]);

            return view;

        }
        //用来存放组件地址的内部类
        static class ViewHolder{
            ImageView iv_icon;
            TextView tv_title;
        }

    }



}

好了,一个最终版的ListView就成型了

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,以下是一个简单的使用自定义适配器实现列表Android代码示例: 1. 首先,我们需要创建一个布局文件,用来定义列表项的显示样式。比如,我们可以创建一个名为list_item.xml的布局文件,定义一个简单的TextView用于显示列表项的内容: ```xml <TextView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/textView" android:layout_width="match_parent" android:layout_height="wrap_content" android:padding="10dp" android:textSize="16sp" /> ``` 2. 接下来,我们需要创建一个自定义适配器类,用于将数据源中的数据绑定到列表项上。比如,我们可以创建一个名为MyAdapter的适配器类,继承自BaseAdapter类,并实现其中的几个方法: ```java public class MyAdapter extends BaseAdapter { private Context mContext; private String[] mData; public MyAdapter(Context context, String[] data) { mContext = context; mData = data; } @Override public int getCount() { return mData.length; } @Override public Object getItem(int position) { return mData[position]; } @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View convertView, ViewGroup parent) { ViewHolder holder; if (convertView == null) { convertView = LayoutInflater.from(mContext).inflate(R.layout.list_item, parent, false); holder = new ViewHolder(); holder.textView = convertView.findViewById(R.id.textView); convertView.setTag(holder); } else { holder = (ViewHolder) convertView.getTag(); } holder.textView.setText(mData[position]); return convertView; } private static class ViewHolder { TextView textView; } } ``` 以上代码中,我们定义了一个MyAdapter类,并在其中实现了BaseAdapter类中的几个重要方法。其中,getCount()方法用于获取数据源中的数据数量,getItem()方法用于获取数据源中指定位置的数据项,getItemId()方法用于获取指定位置的数据项的ID。最重要的是getView()方法,它用于将数据源中的数据绑定到列表项上,并返回一个View对象。 在getView()方法中,我们首先判断convertView是否为空。如果为空,说明当前列表项还没有被创建过,我们就需要通过LayoutInflater从布局文件中加载一个新的View对象,并将其缓存起来。这样,在下次需要显示同样的列表项时,就可以直接从缓存中取出View对象,而不需要重新创建。如果convertView不为空,说明当前列表项已经被创建过,我们可以直接从缓存中取出它的View对象。 接着,我们通过ViewHolder模式,将列表项中的TextView控件缓存起来,避免在每次getView()方法被调用时都需要通过findViewById()方法来查找它。最后,我们将数据源中当前位置的数据绑定到TextView控件上,并返回这个View对象。 3. 最后,我们在Activity中使用自定义适配器,并将它设置到ListView中: ```java ListView listView = findViewById(R.id.listView); String[] data = {"Item 1", "Item 2", "Item 3", "Item 4", "Item 5"}; MyAdapter adapter = new MyAdapter(this, data); listView.setAdapter(adapter); ``` 以上代码中,我们创建了一个MyAdapter适配器对象,并将其设置到ListView中。此时,ListView就会根据适配器中的数据源,创建了对应数量的列表项,并使用自定义的布局文件来显示每个列表项的内容。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值