概念
ContentProvider作为四大组件之一,他的地位不容忽视。它的作用是为不同应用程序数据共享,提供统一接口 我们其他应用程序要调用android系统中的应用,这时候就要用到ContentProvider,也可以我们利用ContentProvider编写自己的内容提供者,供其他应用使用
ContentProvider如何实现数据共享的
ContentProvider通过URI唯一标识其他要访问的数据,开发人员一般通过ContentResolver来进行操作ContentProvider,可进行增删改查实现对数据的共享。
ContentProvider讲解
ContentProvider是一个抽象类,如果要开发自己的内容提供者都要继承这个类并进行重写。主要实现以下方法:
public boolean onCreate()
在创建ContentProvider的时候进行调用
public Cursor query()
用来查询指定的URI返回一个Cursor
public Uri insert()
向指定RUI的ContentProvider中插入数据
public int update()
用户更新指定的RUI的数据
public int delete()
用户删除指定的RUI的数据
public String getType()
用于返回指定的Uri中的数据MIME类型
自定义ContentProvider讲解
我们创建一个数据库的ContentProvider类来实现数据共享,自定义名称为MyProvider
package com.haolin.contentprovider.sample;
import android.content.ContentProvider;
import android.content.ContentValues;
import android.content.Context;
import android.content.UriMatcher;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.net.Uri;
public class MyProvider extends ContentProvider {
private Context mContext;
DBHelper mDbHelper = null;
SQLiteDatabase db = null;
public static final String AUTOHORITY = "com.haolin.contentprovider.sample.MyProvider";
// 设置ContentProvider的唯一标识
public static final int User_Code = 1;
public static final int Job_Code = 2;
// UriMatcher类使用:在ContentProvider 中注册URI
private static final UriMatcher mMatcher;
static{
mMatcher = new UriMatcher(UriMatcher.NO_MATCH);
// 初始化
mMatcher.addURI(AUTOHORITY,"user", User_Code);
mMatcher.addURI(AUTOHORITY, "job", Job_Code);
}
// 以下是ContentProvider的6个方法
/**
* 初始化ContentProvider
*/
@Override
public boolean onCreate() {
mContext = getContext();
// 在ContentProvider创建时对数据库进行初始化
// 运行在主线程,故不能做耗时操作,此处仅作展示
mDbHelper = new DBHelper(getContext());
db = mDbHelper.getWritableDatabase();
// 初始化两个表的数据(先清空两个表,再各加入一个记录)
db.execSQL("delete from user");
db.execSQL("insert into user values(1,'Carson');");
db.execSQL("insert into user values(2,'Kobe');");
db.execSQL("delete from job");
db.execSQL("insert into job values(1,'Android');");
db.execSQL("insert into job values(2,'iOS');");
return true;
}
/**
* 添加数据
*/
@Override
public Uri insert(Uri uri, ContentValues values) {
// 根据URI匹配 URI_CODE,从而匹配ContentProvider中相应的表名
// 该方法在最下面
String table = getTableName(uri);
// 向该表添加数据
db.insert(table, null, values);
// 当该URI的ContentProvider数据发生变化时,通知外界(即访问该ContentProvider数据的访问者)
mContext.getContentResolver().notifyChange(uri, null);
// // 通过ContentUris类从URL中获取ID
// long personid = ContentUris.parseId(uri);
// System.out.println(personid);
return uri;
}
/**
* 查询数据
*/
@Override
public Cursor query(Uri uri, String[] projection, String selection,
String[] selectionArgs, String sortOrder) {
// 根据URI匹配 URI_CODE,从而匹配ContentProvider中相应的表名
// 该方法在最下面
String table = getTableName(uri);
// 查询数据
return db.query(table,projection,selection,selectionArgs,null,null,sortOrder,null);
}
/**
* 更新数据
*/
@Override
public int update(Uri uri, ContentValues values, String selection,
String[] selectionArgs) {
// 由于不展示,此处不作展开
return 0;
}
/**
* 删除数据
*/
@Override
public int delete(Uri uri, String selection, String[] selectionArgs) {
// 由于不展示,此处不作展开
return 0;
}
@Override
public String getType(Uri uri) {
// 由于不展示,此处不作展开
return null;
}
/**
* 根据URI匹配 URI_CODE,从而匹配ContentProvider中相应的表名
*/
private String getTableName(Uri uri){
String tableName = null;
switch (mMatcher.match(uri)) {
case User_Code:
tableName = DBHelper.USER_TABLE_NAME;
break;
case Job_Code:
tableName = DBHelper.JOB_TABLE_NAME;
break;
}
return tableName;
}
}
- ContentProvider作为四大组件之一,我们必须再AndroidManifest.xml文件中进行注册
<provider
android:authorities="com.haolin.contentprovider.sample.MyProvider"
android:name=".MyProvider"
android:exported="true"/>
这里的authorities唯一标识该内容提供者,这样其它的应用才可以找到该内容提供者并操作它的数据;exported为true当前内容提供者可以被其它应用使用,默认为true
操作ContentProvider里的数据
我们在MainActivity中来操作ContentProvider数据 或者新建个项目也可直接操作
/*
* 对user表进行操作
*/
// 设置URI
Uri uri_user = Uri.parse("content://com.haolin.contentprovider.sample.MyProvider/user");
// 插入表中数据
ContentValues values = new ContentValues();
values.put("_id", 3);
values.put("name", "Iverson");
// 获取ContentResolver
ContentResolver resolver = getContentResolver();
// 通过ContentResolver 根据URI 向ContentProvider中插入数据
resolver.insert(uri_user,values);
// 通过ContentResolver 向ContentProvider中查询数据
Cursor cursor = resolver.query(uri_user, new String[]{"_id","name"}, null, null, null);
assert cursor != null;
while (cursor.moveToNext()){
System.out.println("query book:" + cursor.getInt(0) +" "+ cursor.getString(1));
// 将表中数据全部输出
}
cursor.close();
// 关闭游标
/*
* 对job表进行操作
*/
// 和上述类似,只是URI需要更改,从而匹配不同的URI CODE,从而找到不同的数据资源
Uri uri_job = Uri.parse("content://com.haolin.contentprovider.sample.MyProvider/job");
// 插入表中数据
ContentValues values2 = new ContentValues();
values2.put("_id", 3);
values2.put("job", "NBA Player");
// 获取ContentResolver
ContentResolver resolver2 = getContentResolver();
// 通过ContentResolver 根据URI 向ContentProvider中插入数据
resolver2.insert(uri_job,values2);
// 通过ContentResolver 向ContentProvider中查询数据
Cursor cursor2 = resolver2.query(uri_job, new String[]{"_id","job"}, null, null, null);
assert cursor2 != null;
while (cursor2.moveToNext()){
System.out.println("query job:" + cursor2.getInt(0) +" "+ cursor2.getString(1));
// 将表中数据全部输出
}
cursor2.close();
// 关闭游标
这样我们可以对ContentProvider进行增删改查等操作。
ContentObserver
注册ContentObserver来监听对应uri的数据变化,这步不是必须的,如果不需要监听数据变化也可以不注册
首先获取ContentResolver对象
getContentResolver()
在进行注册Observer
resolver.registerContentObserver(uri_user,true,new TestContentObs(handler));
我们需要创建个类继承ContentObserver
package com.haolin.contentprovider.testcontentprovider;
import android.database.ContentObserver;
import android.net.Uri;
import android.os.Handler;
import android.os.Message;
/**
* 作者:haoLin_Lee on 2019/05/12 12:18
* 邮箱:Lhaolin0304@sina.com
* class:
*/
public class TestContentObs extends ContentObserver {
Handler handler;
public TestContentObs(Handler handler) {
super(handler);
this.handler = handler;
}
@Override
public void onChange(boolean selfChange, Uri uri) {
super.onChange(selfChange, uri);
Message message = Message.obtain();
message.obj = uri;
handler.sendMessage(message);
}
}
如果数据库内容有更新,ContentObserver在收到数据变化的通知后通过Handler机制来通知主线程更新UI
感谢大家的阅读
给个赞呗
有问题及时反馈,谢谢大家
关注我 GitHub