文本目录
创建一个工程
初始代码
package com.c20200517.bin0527;
import androidx.appcompat.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.Button;
import android.widget.ListView;
import android.widget.TextView;
import java.util.ArrayList;
import java.util.List;
public class MainActivity extends AppCompatActivity {
ListView listView;//声明ListView
//构造数据结构,用于存填充的数据
List<Chats> lists=new ArrayList();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
listView=findViewById(R.id.listView);
//构造数据
for(int i=0;i<=20;i++){
Chats chat=new Chats();
chat.title="标题"+i;
chat.content="内容"+i;
chat.butext="按钮"+i;
lists.add(chat);
}
listView.setAdapter(new MyAdapter());
}
public class MyAdapter extends BaseAdapter{
@Override
public int getCount() {//填充条目
return lists.size();
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
//在ListView中加载item_layout
View view=View.inflate(MainActivity.this,R.layout.item_layout,null);
Chats chats=lists.get(position);
//获得控件,并填充数据
TextView textView1=view.findViewById(R.id.textView);
TextView textView2=view.findViewById(R.id.textView2);
Button button =view.findViewById(R.id.button);
textView1.setText(chats.title);
textView2.setText(chats.content);
button.setText(chats.butext);
return view;
}
@Override
public Object getItem(int position) {
return null;
}
@Override
public long getItemId(int position) {
return 0;
}
}
}
- 布局文件省略,初始效果如下
存在的问题
1. 如果ListView不去优化,滑倒屏幕以外的item没有回收,那么它所占用的资源依然没有释放,当要构造的条目更多的时候,资源不释放,新的条目又要新的资源,那么MEMORY软件运行所占内存资源越多。
- 滑动屏幕,查看监听器
2. 如果每个条目中还要加载图片资源的话,加载图片是非常耗时的,如果不优化,会给用户造成卡顿,运行慢的感觉。
在以上代码的基础上,开始对ListView进行优化
创建一个类,可在Mactivity里面
1、将item中的findviewbyid保存在一个类
2、需要findviewbyid直接去类里面拿
public class MyViewHolder{
TextView textView1;
TextView textView2;
Button button;
}
找到这些不能被内存回收的item的地址
- 在getView()里面改变为以下代码
//getView()每出现一个item都会调用
//找到这些不能被内存回收的item的地址
//convertView表示系统中没有可回收的item,如果有就返回item的内存地址,没有返回空
@Override
public View getView(int position, View convertView, ViewGroup parent) {
Chats chats=lists.get(position);
MyViewHolder myViewHolder=null;
//屏幕开始时构造的几条item
if(convertView==null){
convertView= LayoutInflater.from(MainActivity.this).inflate(R.layout.item_layout,parent,false);
myViewHolder=new MyViewHolder();
myViewHolder.textView1=convertView.findViewById(R.id.textView);
myViewHolder.textView2=convertView.findViewById(R.id.textView2);
myViewHolder.button=convertView.findViewById(R.id.button);
//MyViewHolder作为convertView的一个成员变量
convertView.setTag(myViewHolder);
}else{
//把作为convertView的成员变量MyViewHolder取出来
myViewHolder=(MyViewHolder) convertView.getTag();
}
//获得控件,并填充数据
myViewHolder.textView1.setText(chats.title);
myViewHolder.textView2.setText(chats.content);
myViewHolder.button.setText(chats.butext);
return convertView;
}
再看效果和资源占用情况
- 效果和未优化一样
- 但资源占用问题却得到优化
完整代码
MainActivity. java
package com.c20200517.bin0527;
import androidx.appcompat.app.AppCompatActivity;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.Button;
import android.widget.ListView;
import android.widget.TextView;
import java.util.ArrayList;
import java.util.List;
public class MainActivity extends AppCompatActivity {
ListView listView;//声明ListView
//构造数据结构,用于存填充的数据
List<Chats> lists=new ArrayList();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
listView=findViewById(R.id.listView);
//构造数据
for(int i=0;i<=100;i++){
Chats chat=new Chats();
chat.title="标题"+i;
chat.content="内容"+i;
chat.butext="按钮"+i;
lists.add(chat);
}
listView.setAdapter(new MyAdapter());
}
public class MyAdapter extends BaseAdapter{
@Override
public int getCount() {//填充条目
return lists.size();
}
//getView()每出现一个item都会调用
//找到这些不能被内存回收的item的地址
//convertView表示系统中没有可回收的item,如果有就返回item的内存地址,没有返回空
@Override
public View getView(int position, View convertView, ViewGroup parent) {
//在ListView中加载item_layout
// View view=View.inflate(MainActivity.this,R.layout.item_layout,null);
Chats chats=lists.get(position);
MyViewHolder myViewHolder=null;
//屏幕开始时构造的几条item
if(convertView==null){
convertView= LayoutInflater.from(MainActivity.this).inflate(R.layout.item_layout,parent,false);
myViewHolder=new MyViewHolder();
myViewHolder.textView1=convertView.findViewById(R.id.textView);
myViewHolder.textView2=convertView.findViewById(R.id.textView2);
myViewHolder.button=convertView.findViewById(R.id.button);
convertView.setTag(myViewHolder);//MyViewHolder作为convertView的一个成员变量
}else{
myViewHolder=(MyViewHolder) convertView.getTag();//把作为convertView的成员变量MyViewHolder取出来
}
myViewHolder.textView1.setText(chats.title);
myViewHolder.textView2.setText(chats.content);
myViewHolder.button.setText(chats.butext);
//获得控件,并填充数据
return convertView;
}
@Override
public Object getItem(int position) {
return null;
}
@Override
public long getItemId(int position) {
return 0;
}
}
//1、将item中的findviewbyid保存在一个类
//2、需要findviewbyid直接去类里面拿
public class MyViewHolder{
TextView textView1;
TextView textView2;
Button button;
}
}
activity_main.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=".MainActivity">
<ListView
android:id="@+id/listView"
android:layout_width="match_parent"
android:layout_height="match_parent"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent">
</ListView>
</androidx.constraintlayout.widget.ConstraintLayout>
item.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:id="@+id/item"
android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView
android:id="@+id/textView"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginStart="24dp"
android:layout_marginLeft="24dp"
android:layout_marginTop="24dp"
android:layout_weight="1"
android:text="TextView"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<TextView
android:id="@+id/textView2"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginTop="32dp"
android:text="TextView"
app:layout_constraintEnd_toEndOf="@+id/textView"
app:layout_constraintStart_toStartOf="@+id/textView"
app:layout_constraintTop_toBottomOf="@+id/textView" />
<Button
android:id="@+id/button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginEnd="100dp"
android:layout_marginRight="32dp"
android:text="Button"
app:layout_constraintBottom_toBottomOf="@+id/textView2"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintTop_toTopOf="@+id/textView" />
</androidx.constraintlayout.widget.ConstraintLayout>
Chats类
package com.c20200517.bin0527;
public class Chats {
String title;
String content;
String butext;
}