删除操作
listItem.remove(position);//选择行的位置
listItemAdapter.notifyDataSetChanged();
list.invalidate();
listview会删除选择的行,重新更新
一、基本用法
(1)由于数组中的数据无法直接传递给ListView,我们需要借助适配器(Adapter)来完成。
(2)ArrayAdapter可以通过泛型来指定要适配的数据,然后在构造函数中把要适配的数据传入。
(3)android.R.layout.simple_list_item_1是ListView内置的一个子项布局,里面只有一个TextView,可显示一段文本
MainActivity中的代码如下:
public class MainActivity extends AppCompatActivity {
private String[] data = {"Sunny","Cloudy",.......,"Unknown"};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//通过泛型来指定要适配的数据类型,然后在构造函数中把适配的数据传入。
//android.R.layout.simple_list_item_1是ListView内置的一个子项布局,里面只有一个TextView,可显示一段文本
//data表示要适配的数据
ArrayAdapter<String> adapter = new ArrayAdapter<String>(MainActivity.this, android.R.layout.simple_list_item_1, data);
ListView listView = (ListView)findViewById(R.id.list_view);
//将构建好的适配器对象传进去
listView.setAdapter(adapter);
}
}
activity_main.xml中的代码如下:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<ListView
android:id="@+id/list_view"
android:layout_width="match_parent"
android:layout_height="match_parent"/>
</LinearLayout>
App截图如下:
需要补充的是:
(1)simple_list_item_1(单行显示),此布局中只有一个TextView,是Android内置的布局文件,id为:android.R.id.text1;
(2)simple_list_item_2和two_line_list_item(双行显示),都有两个TextView:android.R.id.text1和android.R.id.text2,不同之处在于,前者两行字是不一样大的,而后者两行字一样大小。
二、自定义界面
由于只显示一段文本的ListView太过于单调,我们对其界面进行自定义。
首先,定义一个实体类,作为ListView的适配器类型,包含一组天气及其对应的名称。
public class Weather {
private String name;
private int imageId;
public Weather(String name, int imageId) {
this.name = name;
this.imageId = imageId;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getImageId() {
return imageId;
}
public void setImageId(int imageId) {
this.imageId = imageId;
}
}
其次,为ListView的子项指定一个我们自定义的布局,在layout目录下新建weather_item.xml。
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<ImageView
android:id="@+id/weather_image"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/i_999_unknown"/>
<TextView
android:id="@+id/weather_name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:layout_marginLeft="10dp"
android:text="未知天气"/>
</LinearLayout>
然后,创建一个自定义的适配器。这个适配器继承自ArrayAdapter,并将泛型指定为Weather类。
三、提升运行效率
(1)在上一部分中我们在WeatherAdapter类中使用了getView()方法,这个方法在每个子项被滚动到屏幕内的时候会被调用。该方法存在一个弊端:每次都得将布局重新加载一次。当快速滚动ListView的时候,会影响App的性能。getView()方法中还有一个convertView参数,这个参数用于将之前加载好的布局进行缓存,便于重复利用。
我们在getView()中加入判断,如果convertView为null,则使用LayoutInflater去加载布局,不为空则直接重用convertView。
(2)修改之后,虽然不会重复加载布局,但是每次getView()方法还是会调用View的findViewById()方法来获取一次控件的实例。
我们使用ViewHolder来对这部分性能进行优化,ViewHolder通常出现在适配器里,为的是ListView滚动的时候快速设置值,而不必每次都重新创建很多对象,从而提升性能。通过将所有控件的实例都缓存在了ViewHolder里,就不需要每次通过findViewById()方法来获取控件实例。
public class WeatherAdapter extends ArrayAdapter<Weather> {
private int resourceId;
//将上下文、ListView子项布局的id、数据 传递进来
public WeatherAdapter(Context context, int textViewResourceId, List<Weather> obj){
super(context, textViewResourceId, obj);
resourceId = textViewResourceId;
}
@NonNull
@Override
public View getView(int position, @Nullable View convertView, @NonNull ViewGroup parent) {
Weather weather = getItem(position);//获取当前项的Weather实例
//LayoutInflater的inflate()方法接收3个参数:需要实例化布局资源的id、ViewGroup类型视图组对象、false
//false表示只让父布局中声明的layout属性生效,但不会为这个view添加父布局
View view;
ViewHolder viewHolder;
//如果convertView为空,则使用LayoutInflater()去加载布局
if (convertView == null){
view = LayoutInflater.from(getContext()).inflate(resourceId, parent, false);
viewHolder = new ViewHolder();
//通过ViewHolder获取实例
viewHolder.weatherImage = (ImageView) view.findViewById(R.id.weather_image);
viewHolder.weatherName = (TextView) view.findViewById(R.id.weather_name);
//将ViewHolder存储在view中
view.setTag(viewHolder);
}else{
//否则,重用convertView
view = convertView;
//重新获取ViewHolder(利用View的getTag()方法,把ViewHolder重新取出)
viewHolder = (ViewHolder)view.getTag();
}
//设置图片和文字
viewHolder.weatherImage.setImageResource(weather.getImageId());
viewHolder.weatherName.setText(weather.getName());
return view;
}
//定义ViewHolder内部类,用于对控件实例进行缓存
class ViewHolder{
ImageView weatherImage;
TextView weatherName;
}
}
最后,修改MainActivity中的代码。
public class MainActivity extends AppCompatActivity {
private List<Weather> weatherList = new ArrayList<>();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initWeather();//初始化天气数据
WeatherAdapter adapter = new WeatherAdapter(MainActivity.this, R.layout.weather_item, weatherList);
ListView listView = (ListView) findViewById(R.id.list_view);
listView.setAdapter(adapter);
}
private void initWeather(){
//苦力活动,请忽略
Weather i_100 = new Weather("Sunny", R.drawable.i_100_sunny);
weatherList.add(i_100);
Weather i_101 = new Weather("Cloudy", R.drawable.i_101_cloudy);
weatherList.add(i_101);
.............................
}
}
App截图如下:
四、点击事件
修改MainActivity,使用OnItemClickListener()方法为ListView注册监听器。部分代码如下:
//为ListView添加点击事件
listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
Weather weather = weatherList.get(position);
Toast.makeText(MainActivity.this, weather.getName(), Toast.LENGTH_SHORT).show();
}
});
接下来,我们对ListView中的setOnItemClickListener的OnItemClick()方法进行介绍:
//完整版
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {}
//简版
OnItemClick(AdapterView agr0, View arg1, int arg2, long arg3)
举个例子:X,Y两个ListView,X里面有1,2,3,4这4个item,Y里面有a,b,c,d这4个item,如果你点击了b这个item,如下:
(1)arg0,即parent
相当于ListView Y适配器的一个指针,可以通过它来获得Y里面装的一切东西,通俗点就是告诉你,你点的是Y不是X。
(2)arg1,即view
是你点b这个view句柄,就是你可以用这个view,来获得b里的控件的id后操作控件,通过它可以获得该项中的各个组件,例如:arg1.textview.setText(“abc”)。
(3)arg2,即position
是b在Y适配器里的位置(生成ListView时,适配器一个一个的做item,然后把他们按顺序排好队,再放到ListView里,意思就是这个b是第position号做好的)。
(4)arg3,即id
是b在ListView Y里面的第几行,在没有headerView、用户添加的view以及footerView的情况下,position和id的值是一样的。