Android界面中有时候须要显示略微复杂的界面时,就须要我们自己定义一个adapter,而此adapter就要继承BaseAdapter,又一次当中的方法.
Android中Adapter类事实上就是把数据源绑定到指定的View上,然后再返回该View,而返回来的这个View就是ListView中的某一行item。
这里返回来的View正是由我们的Adapter中的getView方法返回的。这样就会easy理解数据是如何一条一条显示在ListView中的。
详细事例:
使用系统带的单选布局
使用系统带的多选布局
使用自己定义布局
数据库我们还使用之前创建的
布局文件:activity_main.java
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingBottom="@dimen/activity_vertical_margin" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" tools:context=".MainActivity" > <ListView android:id="@+id/lv_users" android:layout_width="match_parent" android:layout_height="match_parent" > </ListView> </RelativeLayout>
自己定义布局文件:item_phone
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" > <TextView android:id="@+id/tv_name" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentLeft="true" android:layout_alignParentTop="true" android:layout_marginLeft="40dp" android:layout_marginTop="10dp" android:text="TextView" /> <TextView android:id="@+id/tv_age" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignLeft="@+id/tv_name" android:layout_below="@+id/tv_name" android:layout_marginTop="10dp" android:text="TextView" /> <TextView android:id="@+id/tv_phone" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_below="@+id/tv_name" android:layout_marginLeft="30dp" android:layout_toRightOf="@+id/tv_age" android:text="TextView" /> <TextView android:id="@+id/tv_id" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentLeft="true" android:layout_below="@+id/tv_name" android:paddingLeft="10dp" android:text="1" /> </RelativeLayout>
MainActivity.java
package com.example.android_sqlite; import java.util.List; import android.app.Activity; import android.os.Bundle; import android.view.View; import android.view.ViewGroup; import android.widget.AdapterView; import android.widget.AdapterView.OnItemClickListener; import android.widget.BaseAdapter; import android.widget.CheckedTextView; import android.widget.ListView; import android.widget.TextView; import android.widget.Toast; import com.example.android_sqlite.dao.UserDao; import com.example.android_sqlite.dao.impl.UserDaoImpls; import com.example.android_sqlite.database.DatabaseHelper; import com.example.android_sqlite.domain.Users; /** * BaseAdapter * @author zhaoyazhi * */ public class MainActivity extends Activity implements OnItemClickListener { private ListView lv_users; // 模型层 private List<Users> entites; private UserDao userDao; private DatabaseHelper dh; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); // 设置视图层 lv_users = (ListView) findViewById(R.id.lv_users); // 创建数据库 dh = new DatabaseHelper(this); userDao = new UserDaoImpls(dh); // 获取显示数据 entites = userDao.findAll(); // 创建自己定义的adapter对象 MyAdapyer adapter = new MyAdapyer(); // adapter作为控制层 lv_users.setAdapter(adapter); //注冊点击事件 lv_users.setOnItemClickListener(this); } @Override public void onItemClick(AdapterView<?> parent, View view, int position, long arg3) { //获取点击我的数据 Users object = (Users) parent.getItemAtPosition(position); Toast.makeText(this, object.toString(), 0).show(); } /** * 继承了BaseAdapter 这个BaseAdapter实现了ListAdapter接口 * * @author zhaoyazhi * */ class MyAdapyer extends BaseAdapter { /** * 返回的总条目数量 */ @Override public int getCount() { return entites.size(); } /** * 返回当前条目相应的对象 */ @Override public Object getItem(int position) { return entites.get(position); } /** * */ @Override public long getItemId(int position) { return position; } /** * 返回每一个条目 */ @Override public View getView(int position, View convertView, ViewGroup parent) { return definedItem(position); } private View definedItem(int position) { // 获取当前位置显示的对象 Users entity = entites.get(position); View v = View.inflate(MainActivitys.this, R.layout.item_phone, null); TextView tv_id = (TextView) v.findViewById(R.id.tv_id); TextView tv_name = (TextView) v.findViewById(R.id.tv_name); TextView tv_age = (TextView) v.findViewById(R.id.tv_age); TextView tv_phone = (TextView) v.findViewById(R.id.tv_phone); tv_id.setText(""+entity.getUserId()); tv_name.setText("姓名:" + entity.getUserName()); tv_age.setText("年龄:" + entity.getUserAge()); tv_phone.setText("电话"+entity.getUserPhone()); return v; } /** * 单选框 * * @param position * @return */ private View singleCheck(int position) { // 获取当前位置显示的对象 Users entity = entites.get(position); View v = View.inflate(MainActivitys.this, android.R.layout.simple_list_item_single_choice, null); CheckedTextView tv = (CheckedTextView) v .findViewById(android.R.id.text1); tv.setText(entity.getUserName() + "---" + entity.getUserPhone()); tv.setHeight(40); return tv; } /** * 复选框 * * @param position * @return */ private View multipleCheck(int position) { // 获取当前位置显示的对象 Users entity = entites.get(position); View v = View.inflate(MainActivitys.this, android.R.layout.simple_list_item_multiple_choice, null); CheckedTextView tv = (CheckedTextView) v .findViewById(android.R.id.text1); tv.setText(entity.getUserName() + "---" + entity.getUserPhone()); tv.setHeight(40); return tv; } } }
在完毕这篇文章中的样例之后,我思考了非常长时间,关于重写一个adapter,这当中真的有非常多讲究,遇到一处不懂的都会查阅非常长时间,但也不能保证我已经把当中的重中之重已经找完了,仅仅要你想延伸都能够发现当中的无限.....
问题1:为什么我们要重写一个adapter?
问题2:android中这么多adapter,什么情况下该重写哪一个adapter(ArrayAdapter/SimpleAdapter/SimpleCursorAdapter...)?
问题3:我们重写的adapter为什么是一个内部类,是否建议把adapter做成一个内部类?
问题4:理解应用中的数据源一般都会用一个Map类型的List,有何意途?
问题5:通过adapter是如何做到把一条一条的item放到ListView中的?
问题6:理解重写adapter时,重写的几个方法(getCount()/getItem()/getItemId()/getView())?
问题7:理解ListView使用adapter的机制