Android学习笔记:ListView

删除操作

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的值是一样的。
 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值