Android 系统提供了很多 ContentProvider,以便在应用程序间共享系统数据。 系统提供的 ContentProvider 都存放在 android.provider 包下,例如 android.provider.ContactsContract、android.provider.MediaStore、and
系统提供的 ContentProvider 都存放在 android.provider 包下,例如 android.provider.ContactsContract、android.provider.MediaStore、android.provider.CalendarContract 等。
本节以访问系统联系人列表为例,讲解如何通过系统提供的 ContentProvider 获取数据。
在 Android 2.0(API Level 5)之前,系统所提供的联系人 ContentProvider 为 android.provider.Contacts。
从 Android 2.0 开始,联系人列表相关信息被存放在 android.provider.ContactsContract 中。使用 ContactsContract 获取系统联系人列表的方法与之前有所不同,虽然形式上较以前复杂了一点,但是可以获取一个联系人的多个电话号码。
实例 ContactsCPDemo 演示了使用 ContactsContract 获取系统中所有联系人的名字和电话号码,并且显示出来的过程。为方便起见,假定每个联系人仅有一个电话号码,其运行效果如图 1 所示。
图 1 ContactsCPDemo界面
该效果由 ListView 组件实现。实例 ContactsCPDemo 中布局文件 main.xml 的代码如下:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical"> <TextView android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="联系人列表如下:" /> <ListView android:id="@+id/listView" android:layout_width="fill_parent" android:layout_height="wrap_content" android:padding="5dip" /> </LinearLayout>实例 ContactsCPDemo 要访问系统联系人列表,需要拥有“android.permission.READ_CONTACTS”权限。
实例 ContactsCPDemo 中 AndroidManifest.xml 文件的代码如下:
<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="introduction.android.contactscpdemo"> <uses-permission android:name="android.permission.READ_CONTACTS" /> <application android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name" android:roundIcon="@mipmap/ic_launcher_round" android:supportsRtl="true" android:theme="@style/AppTheme"> <activity android:name=".MainActivity"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> </application> </manifest>实例 ContactsCPDemo 中 MainActivity.java 文件的代码如下:
package introduction.android.contactscpdemo; import java.util.ArrayList; import java.util.HashMap; import java.util.Map; import android.app.Activity; import android.database.Cursor; import android.os.Bundle; import android.provider.ContactsContract; import android.widget.ListView; import android.widget.SimpleAdapter; public class MainActivity extends Activity { private SimpleAdapter listAdapter; private ListView listview; private ArrayList<Map<String, String>> data; private HashMap<String, String> item; /** * Called when the activity is first created. */ @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); listview = (ListView) this.findViewById(R.id.listView); data = new ArrayList<Map<String, String>>(); Cursor cursor = this.getContentResolver().query(ContactsContract.Contacts.CONTENT_URI, null, null, null, null); while (cursor.moveToNext()) { int idFieldIndex = cursor.getColumnIndex(ContactsContract.Contacts._ID); int id = cursor.getInt(idFieldIndex);//根据列名取得该联系人的id int nameFieldIndex = cursor.getColumnIndex(ContactsContract.Contacts.DISPLAY_NAME); String name = cursor.getString(nameFieldIndex); //根据列名取得该联系人的name int numCountFieldIndex = cursor.getColumnIndex (ContactsContract.Contacts.HAS_PHONE_NUMBER); //获取联系人的电话号码个数 int numCount = cursor.getInt(numCountFieldIndex); String phoneNumber = ""; if (numCount > 0) { //联系人至少有一个电话号码 //在类 ContactsContract.CommonDataKinds.Phone中根据id查询相应联系人的所有电话 Cursor phonecursor = getContentResolver().query( ContactsContract.CommonDataKinds.Phone.CONTENT_URI, null, ContactsContract.CommonDataKinds.Phone.CONTACT_ID + "=?", new String[]{Integer.toString(id)}, null); if (phonecursor.moveToFirst()) { //仅读取第一个电话号码 int numFieldIndex = phonecursor.getColumnIndex(ContactsContract. CommonDataKinds.Phone.NUMBER); phoneNumber = phonecursor.getString(numFieldIndex); } } item = new HashMap<String, String>(); item.put("name", name); item.put("phoneNumber", phoneNumber); data.add(item); } listAdapter = new SimpleAdapter(this, data, android.R.layout.simple_list_item_2, new String[]{ "name", "phoneNumber" }, new int[]{android.R.id.text1, android.R.id.text2}); listview.setAdapter(listAdapter); } }其中:
listAdapter = new SimpleAdapter(this, data, android.R.layout.simple_list_item_2, new String[]{ "name", "phoneNumber" }, new int[]{android.R.id.text1, android.R.id.text2}); listview.setAdapter(listAdapter);使用了 Android 系统提供的 simple_list_item_2 布局,并将该布局应用到 main.xml 文件的 ListView 组件中。