一,未优化前
在做作业时写了一个通话记录,其中用到了listview
效果如下:
Mainactivity代码如下:
package com.example.myapplication01;
import androidx.appcompat.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ArrayAdapter;
import android.widget.BaseAdapter;
import android.widget.ListView;
import android.widget.TextView;
import java.util.ArrayList;
import java.util.List;
public class Main5Activity extends AppCompatActivity {
ListView mlistView;
List<String> lists =new ArrayList<>();
List<String> lists2 =new ArrayList<>();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main5);
mlistView =findViewById(R.id.mlistview);
mlistView.setAdapter(new MyAdapter());
for (int i=1;i<50;i++){
if(i%3==0){
lists.add("15"+i%9+"64857864"+" "+"|"+"快递外卖");
}
else if(i%5==0){
lists.add("18"+i%9+"64857864"+" "+"|"+"电话推销");
}
else{
lists.add("13"+i%9+"64857864");
}
}
for (int i=1;i<50;i++){
if(i%4==0){
lists2.add("2019年"+i%11+"月"+i+"日"+" "+"呼出"+i+"秒"+" "+"中国移动");
}
else {
lists2.add("2019年"+i%11+"月"+i+"日"+" "+"呼入"+i+"秒"+" "+"中国联通");
}
}
}
public class MyAdapter extends BaseAdapter{
@Override
public int getCount() {
return lists.size();
}
@Override
public Object getItem(int position) {
return null;
}
@Override
public long getItemId(int position) {
return 0;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
View view = View.inflate(Main5Activity.this,R.layout.home04,null);
String s = lists.get(position);
String s1 = lists2.get(position);
TextView textView = view.findViewById(R.id.textView3);
TextView textView1 = view.findViewById(R.id.textView4);
textView.setText(s);
textView1.setText(s1);
return view;
}
}
}
home04.xml代码如下:
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout 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="wrap_content"
android:background="#FFFFFF"
android:orientation="vertical">
<TextView
android:id="@+id/textView3"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:paddingLeft="30dp"
android:text="TextView"
app:layout_constraintBottom_toTopOf="@+id/textView4"
app:layout_constraintEnd_toStartOf="@+id/button3"
app:layout_constraintHorizontal_bias="0.0"
app:layout_constraintStart_toStartOf="parent" />
<TextView
android:id="@+id/textView4"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:paddingLeft="30dp"
android:text="TextView"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toStartOf="@+id/button3"
app:layout_constraintHorizontal_bias="0.0"
app:layout_constraintStart_toStartOf="parent" />
<Button
android:id="@+id/button3"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginEnd="4dp"
android:layout_marginRight="4dp"
android:background="@drawable/a9"
app:layout_constraintBottom_toBottomOf="@+id/textView4"
app:layout_constraintEnd_toEndOf="parent" />
</androidx.constraintlayout.widget.ConstraintLayout>
mainactivity.xml代码:
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout 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"
tools:context=".Main5Activity"
android:background="#FFFFFF">
<Button
android:id="@+id/button10"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="8dp"
android:layout_marginEnd="30dp"
android:layout_marginRight="30dp"
android:layout_marginBottom="1dp"
android:background="#FFFFFF"
android:text="通话"
android:textColor="#F50707"
app:layout_constraintBottom_toTopOf="@id/mlistview"
app:layout_constraintEnd_toStartOf="@+id/button11"
app:layout_constraintHorizontal_chainStyle="packed"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<Button
android:id="@+id/button11"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="8dp"
android:layout_marginEnd="1dp"
android:layout_marginRight="1dp"
android:background="#FFFFFF"
android:text="联系人"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toEndOf="@+id/button10"
app:layout_constraintTop_toTopOf="parent" />
<ListView
android:id="@+id/mlistview"
android:layout_width="0dp"
android:layout_height="0dp"
android:layout_marginStart="1dp"
android:layout_marginLeft="1dp"
android:layout_marginEnd="1dp"
android:layout_marginRight="1dp"
android:layout_marginBottom="1dp"
android:layout_margin="10dp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/button10" />
</androidx.constraintlayout.widget.ConstraintLayout>
优化后
因为listview不可能要把所有的要显示的内容一下子全部加载到内存,不然会爆,所以listview一般与BaseAdapter共同使用,而Adapter的作用就是将手机屏幕大小最多能显现出的内容加载到内存显现出来,并且通过滑动屏幕滑过多少就加载到内存多少,很大程度上减少内存的使用。我们需要通过一个类继承并重写BaseAdapter
public class MyAdapter extends BaseAdapter{
@Override
public int getCount() {//返回显现内容数据List的大小
return lists.size();
}
@Override
public Object getItem(int position) {
return null;
}
@Override
public long getItemId(int position) {
return 0;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {//加载每条内容到内存并显现的函数
View view = View.inflate(Main5Activity.this,R.layout.home04,null);//加载创造一个界面
String s = lists.get(position);//根据得到的position获得数据
String s1 = lists2.get(position);//获得数据
TextView textView = view.findViewById(R.id.textView3);
TextView textView1 = view.findViewById(R.id.textView4);
textView.setText(s);
textView1.setText(s1);
return View;
}
}
但是后来老师讲解时说这listview还可以进行优化,因为listview将前面的view显现出来,滑出屏幕后,系统并不会进行回收处理,还是很占内存,例如用下面图片说明:
而我们可以通过回收之前创建出来的view,进行二次利用,这样系统只会创建屏幕内的view的个数,减少内存的使用,而在参数中有个convertView,代表的是可以回收利用的view,刚开始时没有滑动listview则没有可利用的,它的值为null,一单滑动则出现可回收利用的view,便不会创建新的view了
实现方法:加个if条件判断convertView是否为null,如果为null则加载view界面,请看注释
代码如下:
public View getView(int position, View convertView, ViewGroup parent) {
// View view = View.inflate(Main5Activity.this,R.layout.home04,null);
if(convertView==null){
convertView= LayoutInflater.from(Main5Activity.this).inflate(R.layout.home04,parent,false);//加载创造一个界面
}
String s = lists.get(position);
String s1 = lists2.get(position);
TextView textView = convertView.findViewById(R.id.textView3);//从回收view中获得TextView
TextView textView1 = convertView.findViewById(R.id.textView4);
convertView.textView.setText(s);
convertView.textView1.setText(s1);
return convertView;
}
二次优化
然而老师说还有在每次convertview.findviewbyid中会多次进行查找,会增加时间复杂度,造成很大的时间开学,所以可以进行优化。
而进行优化的思路就是,将布局文件的属性用一个类进行封装,再将类与每个convertview进行绑定,这样会很好的进行优化。
实现方法:建一个类,Myviewhold,用来封装界面属性
代码如下:
public View getView(int position, View convertView, ViewGroup parent) {
// View view = View.inflate(Main5Activity.this,R.layout.home04,null);未优化前的加载创造界面代码
MyViewHold myViewHold=null;
if(convertView==null){
convertView= LayoutInflater.from(Main5Activity.this).inflate(R.layout.home04,parent,false);
myViewHold =new MyViewHold();
//为该对象添加每个convertview的属性
myViewHold.textView=convertView.findViewById(R.id.textView3);
myViewHold.textView1=convertView.findViewById(R.id.textView4);
//将对象与该convertview进行绑定
convertView.setTag(myViewHold);
}else {
myViewHold = (MyViewHold) convertView.getTag();//获取每个convertview绑定的Myviewhold
}
String s = lists.get(position);
String s1 = lists2.get(position);
//TextView textView = convertView.findViewById(R.id.textView3);未优化前代码
//TextView textView1 = convertView.findViewById(R.id.textView4);未优化前的代码
myViewHold.textView.setText(s);
myViewHold.textView1.setText(s1);
return convertView;
}
}
public static class MyViewHold{//新建一个类,用来封装界面属性
TextView textView;
TextView textView1;
}
}
RecyclerView
补充一下,RecyclerView是listview的加强版,也就是强制优化的listview,下次说一下它