删除操作
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的值是一样的。