ArrayAdapter一个具体的BaseAdapter能支持任意数组对象,在默认情况下这类预期提供的资源id引用的是一个TextView.如果你想使用一个更复杂的布局,使用构造函数也需要一个字段id,该字段id应该引用一个TextView在更大的布局资源。然而对TextView的引用,他会对数组里的每个对象进行toString()方法,你可以添加自定义对象的列表或者数组,覆盖你的对象的toString()方法来确定将显示在项目列表中的文本。用一些除了TextView的控件显示数组,例如ImageView,或者有一些数据除了toString()填充视图,需要覆盖getView()返回你想要的类型。
public class ArrayAdapter<T> extends BaseAdapter implements Filterable {} 继承了BaseAdapter 可以调用其中存在的一些属性方法。实现了Filterable,这里不对其做过多的解释。
构造函数:
1、public ArrayAdapter(Context context, int textViewResourceId) {
init(context, textViewResourceId, 0, new ArrayList<T>());
}
2、 public ArrayAdapter(Context context, int resource, int textViewResourceId) {
init(context, resource, textViewResourceId, new ArrayList<T>());
}
3、 public ArrayAdapter(Context context, int textViewResourceId, T[] objects) {
init(context, textViewResourceId, 0, Arrays.asList(objects));
}
4、public ArrayAdapter(Context context, int resource, int textViewResourceId, T[] objects) {
init(context, resource, textViewResourceId, Arrays.asList(objects));
}
5、 public ArrayAdapter(Context context, int textViewResourceId, List<T> objects) {
init(context, textViewResourceId, 0, objects);
}
6、 public ArrayAdapter(Context context, int resource, int textViewResourceId, List<T> objects) {
init(context, resource, textViewResourceId, objects);
}
这六个构造函数其中最多包括的参数有4个,Context 当前上下文,布局文件(可以省略),textView资源id和 数据来源(数组或者集合),同样都调用了一个init()函数。
private void init(Context context, int resource, int textViewResourceId, List<T> objects) {
mContext = context;
mInflater = (LayoutInflater)context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
mResource = mDropDownResource = resource;
mObjects = objects;
mFieldId = textViewResourceId;
}
分别赋值给类中的私有域,mInflater为LayoutInflater,产生相应项的视图。
mResource = mDropDownResource = resource;显示填充视图内容的资源
类中有两个域保存数据
private List<T> mObjects;
private ArrayList<T> mOriginalValues;
在createViewFromResource中,首先判断convertView是否存在,若不存在则inflate一个,然后判断mFieldID是否为0,若为0则表示传递给ArrayAdapter的资源ID为一TextView,否则是传递了一Layout,mFieldID为此Layout中TextView的ID。然后通过getItem取得相应位置的数据项,判断是否是CharSequence的实例,如果是直接setText,否则调用其toString()函数,可以ArrayAdapter默认只能给TextVext设置字符串,若要使用其他视图,需要重载getView或getDropDownView,一般情况下会继承BaseAdapter自定义自己的Adapter。静态函数 public static ArrayAdapter<CharSequence> createFromResource(Context context,
int textArrayResId, int textViewResId) {
CharSequence[] strings = context.getResources().getTextArray(textArrayResId);
return new ArrayAdapter<CharSequence>(context, textViewResId, strings);
}
读取资源文件中定义的字符数组作为数据生成ArrayAdapter,可以看到只能用TextView视图,而不可以指定一Layout再指定Layout中一个TextView的ID。
在ArrayAdapter中还定义了一个ArrayFilter,继承自Filter,用于过滤数据项(当ListView有焦点时,通过键盘输入字符来进行列表项的过滤),其过滤方法为传入一CharSequence,判断相应数据项是否以此CharSequence开头,若不是则用空格分割些数据项,判断分割后的各字符串是否以此CharSequence开头,若是则保留(若数据不是CharSequence则调用其toString())。如果传入的CharSequence为null或长度为0则不过滤。前面写了3篇关于android的文章,其中的演示程序都写在了一个工程中,当时为了方便测试就在启动页MainActivity中放了3个按钮,点击不同的按钮进入不同的示例程序页面,MainActivity的界面如下截图:
按照上面的设计,每写一个演示程序就需要在MainActivity中新增一个按钮并为其写点击事件,感觉这样有点麻烦,于是想改进一下,顺便用一下ListView和ArrayAdapter。经过改进后的启动页如下图:
新建了一个MainActivity2,用一个ListView来展现演示程序列表,点击相应列表项就跳转到相应的演示页面,功能和之前的MainActivity一样,但之后就不需要不断添加按钮及其点击事件了,只需要为每个演示页面添加一行代码:catalogs.add(new Catalog("演示的名称", 要跳转到的Activity.class)),这样就简洁多了,具体代码如下。
MainActivity2布局文件如下:
<?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"
android:orientation="vertical" >
<ListView
android:id="@+id/lvCatalog"
android:layout_width="fill_parent"
android:layout_height="fill_parent" >
</ListView>
</LinearLayout>MainActivity2 Java代码如下:
package chengyujia.demo.aty;
import java.util.ArrayList;
import java.util.List;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import chengyujia.demo.R;
import chengyujia.demo.model.Catalog;
public class MainActivity2 extends BaseActivity {
private ListView lvCatalog;
private List<Catalog> catalogs;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main2);
lvCatalog = (ListView) findViewById(R.id.lvCatalog);
init();
}
private void init() {
catalogs = new ArrayList<Catalog>();
// 为主页目录添加条目,以后每多写一个例子,在这里添加一个条目就可以了。
// 不像以前MainActivity中每多写一个例子,需要添加一个按钮,还要写相应的点击事件。
catalogs.add(new Catalog("屏幕方向", OrientationActivity.class));
catalogs.add(new Catalog("打电话", CallActivity.class));
catalogs.add(new Catalog("版本信息", VersionActivity.class));
// 这里ListView的适配器选用ArrayAdapter,ListView中每一项的布局选用系统的simple_list_item_1。
ArrayAdapter<Catalog> adapter = new ArrayAdapter<Catalog>(this, android.R.layout.simple_list_item_1, catalogs);
lvCatalog.setAdapter(adapter);
// 通过一个实现OnItemClickListener接口的匿名类的onItemClick方法来处理ListView中每一项的点击事件。
lvCatalog.setOnItemClickListener(new OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
startActivity(new Intent(MainActivity2.this, catalogs.get(position).cls));
}
});
}
}在这里还使用了一个自定义的Model类Catalog,用来表示列表中的每一项绑定的数据,其代码如下:
package chengyujia.demo.model;
import android.app.Activity;
//主页目录列表项
public class Catalog {
// 条目名称
public String name;
// 点击条目要跳转到的目标Activity的类型
public Class<? extends Activity> cls;
// 写个带参数的构造方法,方便赋值使用。
public Catalog(String name, Class<? extends Activity> cls) {
this.name = name;
this.cls = cls;
}
// ListView使用ArrayAdapter<T>时每个列表项显示的内容就是T的toString方法返回的值,
// 如果这里不重写则会调用父类Object的toString方法。
@Override
public String toString() {
return name;
}
}Catalog类中注意的是一定要重写toString方法,不然的话ListView的每一项显示的是“对象的类名+@+hashCode”的组合字符串。
来源:https://www.cnblogs.com/chengyujia/archive/2015/12/08/5031323.html



