Android studio 之 ListView

ListView 绑定的适配器是 BaseAdapter。

// 获取视图(设置 listView 每一项的显示效果)
    /* 参数1:当前Item的下标 --- 和数据源的下标相同,可以由此获取数据源配置item
     * 参数2:当前Item的view
     * 参数3:当前视图的父视图(可调整当前视图的宽高)*/
    public View getView(int position, View convertView, ViewGroup parent) {
        // 完成对view的设置
        // 将设置好的 item 布局资源转换成view
        convertView = LayoutInflater.from(context).inflate(R.layout.baseadapter_item,null); // 此时得到的是最初的item布局,没有添加的数据

        // 获取数据源[position] 的数据,并将他们设置到item视图中的控件中
        BaseMsg m = baseMsgList.get(position); // 获取item的数据列表

        ImageView imageView = convertView.findViewById(R.id.base_icon);
        imageView.setImageResource(m.getIcon());

        TextView tN = convertView.findViewById(R.id.base_name);
        tN.setText(m.getNickName());

        TextView tC = convertView.findViewById(R.id.base_content);
        tC.setText(m.getContent());

        // 可以给item的单个控件设置点击事件
        tN.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
              //  System.out.printf("点击了"+tN.getText());
                Toast.makeText(context,"你点击了"+m.getNickName(),Toast.LENGTH_SHORT).show();
            }
        });
        return convertView;
    }

问题一:优化view对象 

在MyAdapter类中的getView方法中,我们注意到,上面的写法在每个视图出现时都会执行,有多少个item就会调用多少次 getView() 如果item太多就会很浪费资源!!

优化:可以复用convertView

  • 在 getView 方法中,首先检查 convertView 是否为空。
  • 如果为空,表示当前屏幕上没有可复用的 item 视图,需要创建新的视图并进行布局设置。
  • 但是,如果 convertView 不为空,就意味着该视图已经被创建过,并且已经在屏幕上显示过,这时我们可以直接复用 convertView,避免了重复的布局加载和赋值操作
 public View getView(int position, View convertView, ViewGroup parent) {

        // 优化1:利用进入 RecyclerBin 中的view,减少对view的赋值
        /* 当视图第一次构建后,上下滑动到看不见时就会进入 RecyclerBin,此时 convertView 就不为null,
         * 我们就可以复用这个 convertView */
        if( convertView == null){
            // 完成对view的设置
            // 将设置好的 item 布局资源转换成view
            convertView = LayoutInflater.from(context).inflate(R.layout.baseadapter_item,null); // 此时得到的是最初的item布局,没有添加的数据
            System.out.println("当前显示的视图"+(position+1));
        }
        // 获取数据源[position] 的数据,并将他们设置到item视图中的控件中
        BaseMsg m = baseMsgList.get(position); // 获取item的数据列表

        ImageView imageView = convertView.findViewById(R.id.base_icon);
        imageView.setImageResource(m.getIcon());

        TextView tN = convertView.findViewById(R.id.base_name);
        tN.setText(m.getNickName());

        TextView tC = convertView.findViewById(R.id.base_content);
        tC.setText(m.getContent());

        // 可以给item的单个控件设置点击事件
        tN.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
              //  System.out.printf("点击了"+tN.getText());
                Toast.makeText(context,"你点击了"+m.getNickName(),Toast.LENGTH_SHORT).show();
            }
        });
        return convertView;
    }

问题二:优化view中组件的查找 

即使经过优化1的步骤,可以减少view的构建,但是每出现一个view已经会寻找一次控件id,findViewById 依旧十分浪费资源,因此可以进行第二步优化

优化:利用viewHolder

在convertView为null的时候,我们不仅重新inflate出来一个view,并且还需要进行findviewbyId的查找工作,但是同时我们还需要获取一个ViewHolder类的对象,并将findviewById的结果赋值给ViewHolder中对应的成员变量。最后将holder对象与该view对象“绑”在一块--setTag方法

当convertView不为null时,我们让view=converView,同时取出这个view对应的holder对象--getTag方法,就获得了这个view对象中的TextView组件,

// 即使经过优化1的步骤,可以减少view的构建,但是每出现一个view已经会寻找一次控件id,findViewById 依旧十分浪费资源,因此可以进行第二步优化
// 1.定义一个 内部类 ViewHolder,并声明需要用到的控件属性
// 2.当 convertView == null 当前视图没有被创建时,初始化一个 ViewHolder 对象,并为其中的控件赋值
// 3.调用 contvertView.setTag(ViewHolder) 将这个 HolderView 绑定到这个view中
// 4.当 convertView != null时,通过 contvertView.getTag() 找到绑定的ViewHolder
       
ViewHolder holder;
if (convertView == null) {
  // 完成对view的设置
  // 将设置好的 item 布局资源转换成view
  convertView = LayoutInflater.from(context).inflate(R.layout.baseadapter_item, null); // 此时得到的是最初的item布局,没有添加的数据

   holder = new ViewHolder();
   holder.icon = convertView.findViewById(R.id.base_icon);
   holder.nickName = convertView.findViewById(R.id.base_name);
   holder.content = convertView.findViewById(R.id.base_content);
   convertView.setTag(holder);
   System.out.println("当前显示的视图" + (position + 1));
} else {
   holder = (ViewHolder) convertView.getTag(); // 取出view中绑定的ViewHolder,此时要强转成 ViewHolder 类型
 }
 public View getView(int position, View convertView, ViewGroup parent) {

        // 优化1:利用进入 RecyclerBin 中的view,减少对view的赋值
        /* 当视图第一次构建后,上下滑动到看不见时就会进入 RecyclerBin,此时 convertView 就不为null,
         * 我们就可以复用这个 convertView */

        // 即使经过优化1的步骤,可以减少view的构建,但是每出现一个view已经会寻找一次控件id,findViewById 依旧十分浪费资源,因此可以进行第二步优化
        // 1.定义一个 内部类 ViewHolder,并声明需要用到的控件属性
        // 2.当 convertView == null 当前视图没有被创建时,初始化一个 ViewHolder 对象,并为其中的控件赋值
        // 3.调用 contvertView.setTag(ViewHolder) 将这个 HolderView 保存到这个view中
        // 4.当 convertView != null时,通过 contvertView.getTag() 找到保存的ViewHolder
        ViewHolder holder;
        if (convertView == null) {
            // 完成对view的设置
            // 将设置好的 item 布局资源转换成view
            convertView = LayoutInflater.from(context).inflate(R.layout.baseadapter_item, null); // 此时得到的是最初的item布局,没有添加的数据

            holder = new ViewHolder();
            holder.icon = convertView.findViewById(R.id.base_icon);
            holder.nickName = convertView.findViewById(R.id.base_name);
            holder.content = convertView.findViewById(R.id.base_content);
            convertView.setTag(holder);

            System.out.println("当前显示的视图" + (position + 1));
        } else {
            holder = (ViewHolder) convertView.getTag(); // 此时要强转成 ViewHolder 类型
        }
        // 获取数据源[position] 的数据,并将他们设置到item视图中的控件中
        BaseMsg m = baseMsgList.get(position); // 获取item的数据列表

        /*ImageView imageView = convertView.findViewById(R.id.base_icon);
        imageView.setImageResource(m.getIcon());*/
        holder.icon.setImageResource(m.getIcon());

        /*TextView tN = convertView.findViewById(R.id.base_name);
        tN.setText(m.getNickName());*/
        holder.nickName.setText(m.getNickName());

        /*TextView tC = convertView.findViewById(R.id.base_content);
        tC.setText(m.getContent());*/
        holder.content.setText(m.getContent());

        // 可以给item的单个控件设置点击事件
        holder.nickName.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Toast.makeText(context, "你点击了" + m.getNickName(), Toast.LENGTH_SHORT).show();
            }
        });
        return convertView;
    }