Android开发学习之路--Content Provider之初体验
天气说变就变,马上又变冷了,还好空气不错,阳光也不错,早起上班的车上的人也不多,公司来的同事和昨天一样一样的,可能明天会多一些吧,那就再来学习android吧。学了两个android的组件,这里学习下第三个android的组件,Content Provider内容提供器。
Content Provider向我们提供了在不同应用程序之间的数据共享,比如微信啊,支付宝啊,想要获取手机联系人的信息,而手机联系人是另一个应用程序,那么这时候就需要用到Content Provider了。Content Provider为存储和获取数据提供了统一的接口,对数据进行了封装,我们不用关心数据存储的细节,使用表的形式来组织数据。Android提供了一些默认的ContentProvider,比如音频,视频,图片和通讯录等。
类似于文件,SharedPreferences或者SQLiteDataBase都有一个路径,那么ContentProvider也是有路径的,它的路径由Uri表示,Uri主要包含了两部分的信息,一个是需要操作的ContentProvider,另一个就是对ContentProvider中的什么数据进行操作。一个简单的Uri:
content://com.example.contentprovidertest.provider/person/2
scheme:content://表示,这个是android定义好的,不能改变的
主机名或者authority:com.example.contentprovidertest.provider表示,唯一的,一般指的是包名,
路径:person,表示的是person这个表,2表示id为2。
如果要把上面的字符串转变为uri,那么就要使用Uri类中的parse()方法,如下:
Uri uri=Uri.parse("content://com.example.contentprovidertest.provider/person/2");
这里先来写个例子,用来获取通讯录中的信息吧。新建ContentProviderTest工程,因为模拟器的联系人还没有创建,这里先添加几个好友:

接着我们继续编写代码,先在layout中添加listview,因为通讯录一般都是用listview来实现的。代码如下:
<!--{cke_protected}{C}%3C!%2D%2D%3Fxml%20version%3D%221.0%22%20encoding%3D%22utf-8%22%3F%2D%2D%3E-->
<linearlayout 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:orientation="vertical" tools:context="com.example.jared.contentprovidertest.MainActivity">
<listview android:id="@+id/contacts" android:layout_height="match_parent" android:layout_width="match_parent">
</listview></linearlayout>
这里就简单地加了一个listview,接着修改MainAcitivity代码:
package com.example.jared.contentprovidertest;
import android.database.Cursor;
import android.os.Bundle;
import android.provider.ContactsContract;
import android.support.v7.app.AppCompatActivity;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import java.util.ArrayList;
import java.util.List;
public class MainActivity extends AppCompatActivity {
private ListView contactViews;
ArrayAdapter adapter;
List contactList = new ArrayList();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
contactViews = (ListView)findViewById(R.id.contacts);
adapter = new ArrayAdapter(this, android.R.layout.simple_list_item_1, contactList);
contactViews.setAdapter(adapter);
readContacts();
}
public void readContacts() {
Cursor cursor = null;
try {
cursor = getContentResolver().query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI,
null, null, null, null);
while(cursor.moveToNext()) {
String ContactName = cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME));
String ContactPhoneNum = cursor.getString(cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER));
contactList.add(ContactName+'\n'+ContactPhoneNum);
}
} catch (Exception e) {
e.printStackTrace();
} finally {
if(cursor != null){
cursor.close();
}
}
}
}
这里用了string类型的适配器,然后通过cursor来获取联系人名字和电话号码,其中的名字和电话的定义最终也是字符串。
String DISPLAY_NAME = "display_name";
String NUMBER = "number";这个和数据库的存取很类似,接着在AndroidManifest中添加权限:
然后运行app,效果如下:

由上可知,获得了三个联系人的信息,也就是我们刚保存的信息。
接着学习自定义的ContentProvider,这里借用了dbtest中的MyDBHelper类,新建myContenProvider类,代码如下:
package com.example.jared.contentprovidertest;
import android.content.ContentProvider;
import android.content.ContentValues;
import android.content.UriMatcher;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.net.Uri;
import android.support.annotation.Nullable;
/**
* Created by jared on 16/2/15.
*/
public class myContentProvider extends ContentProvider {
public static final int PERSON_DIR = 0;
public static final int PERSON_ITEM = 1;
public static final int TEACHER_DIR = 2;
public static final int TEACHER_ITEM = 3;
private static UriMatcher uriMatcher;
private static final String AUTHORITY = "com.example.jared.contentprovidertest.provider";
private MyDBHelper myDBHelper;
static {
uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
uriMatcher.addURI(AUTHORITY, "person", PERSON_DIR);
uriMatcher.addURI(AUTHORITY, "person/#", PERSON_ITEM);
uriMatcher.addURI(AUTHORITY, "teacher", TEACHER_DIR);
uriMatcher.addURI(AUTHORITY, "teacher/#", TEACHER_DIR);
}
@Override
public boolean onCreate() {
myDBHelper = new MyDBHelper(getContext(), "PersonStore.db", null, 2);
return true;
}
@Override
public int update(Uri uri, ContentValues contentValues, String s, String[] strings) {
SQLiteDatabase db = myDBHelper.getWritableDatabase();
int updateRows = 0;
switch (uriMatcher.match(uri)) {
case PERSON_DIR:
updateRows = db.update("person", contentValues, s, strings);
break;
case PERSON_ITEM:
String personId = uri.getPathSegments().get(1);
updateRows = db.update("person", contentValues, "id = ?", new String[]{personId});
break;
case TEACHER_DIR:
updateRows = db.update("teacher", contentValues, s, strings);
break;
case TEACHER_ITEM:
String teacherId = uri.getPathSegments().get(1);
updateRows = db.update("teacher", contentValues, "id = ?", new String[]{teacherId});
break;
default:
break;
}
return updateRows;
}
@Override
public int delete(Uri uri, String s, String[] strings) {
SQLiteDatabase db = myDBHelper.getWritableDatabase();
int deleteRows = 0;
switch (uriMatcher.match(uri)) {
case PERSON_DIR:
deleteRows = db.delete("person", s, strings);
break;
case PERSON_ITEM:
String personId = uri.getPathSegments().get(1);
deleteRows = db.delete("person", "id = ?", new String[]{personId});
break;
case TEACHER_DIR:
deleteRows = db.delete("teacher", s, strings);

