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; }