ContentProvider 使用示例

contentProvider是Android中提供的专门用于不同应用间进行数据共享方式,作为四大组件之一的contentProvider,使用起来比较简单,但是因为使用频率不是很高,所以经常遗忘其使用方法,所以在次记录下其基本用法的实例。


首先,有必要对contentProvider的基本原理进行一个简单的讨论。
contentProvider的底层实现是通过Binder实现,但是他的使用比aidl要简单很多,因为系统已经为我们做了很多的封装。
系统中也为我们提供了非常多的ContentProvider,比如通讯录,日程表,通话记录等,要跨进程访问这些信息,只需要通过ContentResolver的query、update、insert、update、insert、delete方法即可。
要实现自定义ContentProvider,继承ContentProvider类并实现六个抽象方法即可,这六个抽象方法分别是是:OnCreate、query、update、insert、delete和getType。需要注意的是在这六个方法中,除了OnCreate由系统回调并运行在主线程里,其他五个方法均由外界回调并运行在Binder线程池中。

下面我们自定义一个BookProvider继承自ContentProvider,其代码如下:

public class BookProvider extends ContentProvider {

public static final String TAG = "reoger.hut.hello.word";

public static final String AUTHORLTY = "reoger.hut.hello.word";

public static final Uri BOOK_CONTENT_URI = Uri.parse("content://"+AUTHORLTY+"/book");
public static final Uri USER_CONTENT_URI = Uri.parse("content://"+AUTHORLTY+"/user");

public static final int BOOK_URI_CODE = 0;
public static final int USER_URI_CODE = 1;

private static final UriMatcher sUriMatcher = new UriMatcher(UriMatcher.NO_MATCH);

private Context mContext;
private SQLiteDatabase mDb;

static {
sUriMatcher.addURI(AUTHORLTY,"book",BOOK_URI_CODE);
sUriMatcher.addURI(AUTHORLTY,"user",USER_URI_CODE);
}

private String getTableName(Uri uri){
String tableName = null;
switch (sUriMatcher.match(uri)){
case BOOK_URI_CODE:
tableName = DbOpenHelper.BOOK_TABLE_NAME;
break;
case USER_URI_CODE:
tableName = DbOpenHelper.USER_TABLE_NAME;
break;
default:
break;
}
return tableName;
}

@Override
public boolean onCreate() {
Log.d(TAG, "onCreate: " + Thread.currentThread());
mContext = getContext();
initProvider();
return true;
}

/**
* 初始化数据库
*/
private void initProvider() {
mDb = new DbOpenHelper(mContext).getWritableDatabase();
mDb.execSQL("delete from "+DbOpenHelper.BOOK_TABLE_NAME);
mDb.execSQL("delete from "+DbOpenHelper.USER_TABLE_NAME);
mDb.execSQL("insert into book values(3,'android');");
mDb.execSQL("insert into book values(4,'ios');");
mDb.execSQL("insert into book values(5,'html5');");
mDb.execSQL("insert into user values(1,'jack',1);");
mDb.execSQL("insert into user values(2,'jany',0);");

}

@Nullable
@Override
public Cursor query(@NonNull Uri uri, @Nullable String[] projection, @Nullable String selection, @Nullable String[] selectionArgs, @Nullable String sortOrder) {
Log.d(TAG, "query: " + Thread.currentThread().getName());

String table = getTableName(uri);
if(table == null){
throw new IllegalArgumentException(" UnSupported URL: "+uri);
}
return mDb.query(table,projection,selection,selectionArgs,null,null,sortOrder,null);
}

@Nullable
@Override
public String getType(@NonNull Uri uri) {
return null;
}

@Nullable
@Override
public Uri insert(@NonNull Uri uri, @Nullable ContentValues values) {
Log.d(TAG, "insert: ");

String table = getTableName(uri);
if(table == null ){
throw new IllegalArgumentException("UnSupported URI :"+uri);
}
mDb.insert(table,null,values);
mContext.getContentResolver().notifyChange(uri,null);
return uri;
}

@Override
public int delete(@NonNull Uri uri, @Nullable String selection, @Nullable String[] selectionArgs) {
Log.d(TAG, "delete: ");
String table = getTableName(uri);
if(table == null ){
throw new IllegalArgumentException("UnSupported URI :"+uri);
}
int count = mDb.delete(table,selection,selectionArgs);
if(count>0){
getContext().getContentResolver().notifyChange(uri,null);
}
return count;
}

@Override
public int update(@NonNull Uri uri, @Nullable ContentValues values, @Nullable String selection, @Nullable String[] selectionArgs) {
Log.d(TAG, "update: ");
String table = getTableName(uri);
if(table == null ){
throw new IllegalArgumentException("UnSupported URI :"+uri);
}
int row = mDb.update(table,values,selection,selectionArgs);
if(row >0 ){
getContext().getContentResolver().notifyChange(uri,null);
}
return row;
}
}

因为其共享的内容通过数据库存储,所以需要一个数据库的帮助类。用来帮我们创建数据库,代码很简单,DbOpenHelper的代码如下:

public class DbOpenHelper extends SQLiteOpenHelper {

private static final String DB_NAME = "boook_provider.db";
public static final String BOOK_TABLE_NAME = "book";
public static final String USER_TABLE_NAME = "user";

private static final int DB_VERSION = 1;

private String CREATE_BOOK_TABLE = "CREATE TABLE IF NOT EXISTS "
+ BOOK_TABLE_NAME + "( _id INTEGER PRIMARY KEY," + "name TEXT)";
private String CREATE_USER_TABLE = "CREATE TABLE IF NOT EXISTS "
+ USER_TABLE_NAME + "( _id INTEGER PRIMARY KEY," + "name TEXT ,"+"sex INT)";

public DbOpenHelper(Context context) {
super(context, DB_NAME, null, DB_VERSION);
}

@Override
public void onCreate(SQLiteDatabase db) {
db.execSQL(CREATE_BOOK_TABLE);
db.execSQL(CREATE_USER_TABLE);
}

@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {

}
}

需要说明的是,CongetProvider可以共享的数据不仅只是数据库的,也可以共享文件中的数据,甚至可以共享内存中的数据。这里只是通过共享数据库中的内容作为示例 。
当然,作为android四大组件,都是需要在manifest中进行申明(当然,broadcast也可以在java代码中进行声明)。所以这里BookProvider声明的代码如下:

<provider
android:name=".BookProvider"
android:authorities="reoger.hut.hello.word"
android:permission="com.reoger.hello"
android:process=":provider"
/>

其中authorities属性就声明了外界访问这个ContentProvider的Uri。例如此处访问的URI的值应该是:content://reoger.hut.hello.word
通过permission属性为provider定制权限,需要访问此provider必须声明此处定义的permission,否则无法正常获取到信息。当然,为了让其运行在一个单独的进程中,指定了process属性。

通过前面的代码,自定义的ContentProvider已经实现完毕,接下来我们来检验自定义的BookProvider是否正常工作。
MainActivity代码如下;

public class MainActivity extends AppCompatActivity {

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);

Uri uri = Uri.parse("content://reoger.hut.hello.word/book");

ContentValues values = new ContentValues();
values.put("_id",6);
values.put("name","Android 开发艺术");
getContentResolver().insert(uri,values);

Cursor bookCursor = getContentResolver().query(uri,new String[]{"_id","name"},null,null,null);
while(bookCursor.moveToNext()){
Book book = new Book(bookCursor.getInt(0),bookCursor.getString(1));
Log.d("TAG","bookID "+book.toString());
}
bookCursor.close();

Uri uri2 = Uri.parse("content://reoger.hut.hello.word/user");

ContentValues values2 = new ContentValues();
values2.put("_id",3);
values2.put("name","reoger");
values.put("sex","0");
getContentResolver().insert(uri2,values2);

Cursor bookCursor2 = getContentResolver().query(uri2,new String[]{"_id","name","sex"},null,null,null);
while(bookCursor2.moveToNext()){
User user = new User(bookCursor2.getInt(0),bookCursor2.getString(1),bookCursor2.getInt(2));
Log.d("TAG","bookID "+user.toString());
}
bookCursor2.close();
}
}

通过打印的日志信息,发现我们的测试示例完全符合我们的期望。结果如图所示:

15-50-29.jpg

最后,补充一下其中Book和User的代码:

public class Book implements Parcelable{
public Book(int id, String name) {
this.id = id;
this.name = name;
}

private int id;
private String name;

public Book() {
}

protected Book(Parcel in) {
id = in.readInt();
name = in.readString();
}

public static final Creator<Book> CREATOR = new Creator<Book>() {
@Override
public Book createFromParcel(Parcel in) {
return new Book(in);
}

@Override
public Book[] newArray(int size) {
return new Book[size];
}
};

@Override
public int describeContents() {
return 0;
}

@Override
public void writeToParcel(Parcel dest, int flags) {
dest.writeInt(id);
dest.writeString(name);
}

@Override
public String toString() {
String result = "book = ["+id+
","+name+"];";
return result;
}
}

public class User implements Parcelable {

private int id;
private String name;
private int sex;

public User(int id, String name, int sex) {
this.id = id;
this.name = name;
this.sex = sex;
}

@Override
public String toString() {
return "id= "+id+" ,name ="+name+" ,sex= "+sex;
}

protected User(Parcel in) {
id = in.readInt();
name = in.readString();
sex = in.readInt();
}

public static final Creator<User> CREATOR = new Creator<User>() {
@Override
public User createFromParcel(Parcel in) {
return new User(in);
}

@Override
public User[] newArray(int size) {
return new User[size];
}
};

@Override
public int describeContents() {
return 0;
}

@Override
public void writeToParcel(Parcel dest, int flags) {
dest.writeInt(id);
dest.writeString(name);
dest.writeInt(sex);
}
}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,293评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,604评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,958评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,729评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,719评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,630评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,000评论 3 397
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,665评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,909评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,646评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,726评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,400评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,986评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,959评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,996评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,481评论 2 342

推荐阅读更多精彩内容