续上一博文(Android核心基础-5.Android 数据存储与访问-3.使用Sqlite进行数据存储)
四、ContentProvider 内容提供者
4.1 什么是ContentProvider
ContentProvider是安卓四大组件之一, 用来共享应用程序内的数据
该组件对外提供了其他应用可以直接访问的增删改查方法
在数据被修改的时候, 可以使用ContentObserver监听
4.2 创建ContentProvider***
定义类继承ContentProvider
在清单文件中声明< provider>标签
4.3 访问ContentProvider***
获取ContentResolver对象
使用ContentResolver指定Uri即可对指定的ContentProvider增删改查
4.4 增删改查方法*
ContentProvider的insert(), delete(), update(), query(): 对外提供的4个操作数据的方法
ContentResolver的insert(), delete(), update(), query(): 调用ContentProvider的方法
SQLiteDabase的insert(), delete(), update(), query(): 在ContentProvider中适合用这4个方法操作数据库, 其内部就是拼接SQL语句, 调用execSQL()和rawQuery()
4.5 UriMatcher*
UriMatcher可以用来匹配Uri, 识别出子级路径
addUri()方法可以指定路径和结果码
match()方法可以匹配一个Uri, 得到结果码
4.6 带id的Uri*
可以使用UriMatcher添加一个带”#”的路径, 用来匹配带id的Uri
使用ContentUris.parseId()可以从Uri中解析出id
4.7 ContentObserver监听数据修改**
可以使用ContentResolver, 调用registerContentObserver()注册一个ContentObserver
在数据修改时使用ContentResolver调用notifyChange()发一个通知
ContentObserver会收到这个通知, 执行内部的onChange()方法
发送通知:
// 当数据改变时, 发送通知, 这时ContentObserver就会接收到 getContext().getContentResolver().notifyChange(uri, null);
监听通知:
// 注册一个ContentObserver(类似一个监听器), 其中需要实现onChange()方法, 在数据修改的时候, 会自动执行onChange()方法 Uri uri = Uri.parse("content://net.dxs.provider"); //参数二:是否接收后代通知,如content://net.dxs.provider/account getContentResolver().registerContentObserver(uri, true, new MyObserver()); private class MyObserver extends ContentObserver { public MyObserver() { super(new Handler()); } @Override public void onChange(boolean selfChange) { list = dao.queryPage(20, pageNum); // 重新查询数据 adapter.notifyDataSetChanged(); // 刷新ListView(把数据和ListView显示的内容同步) } }
4.8 监听短信
从github上下载telephonyprovider, 从清单文件中获取Uri
在程序中对指定Uri注册ContentObserver, 当收发短信时就会执行onChange()
查询到最后一条数据就是短信记录
4.9 ContentProvider匹配说明
1. schema,用来说明一个ContentProvider控制这些数据。 “content://”
2. 主机名或授权(Authority),它定义了是哪个ContentProvider提供这些数据。
3. path路径,URI下的某一个Item。
4. ID, 通常定义Uri时使用”#”号占位符代替, 使用时替换成对应的数字
“content://net.dxs.provider/person/#” #表示数据id(#代表任意数字)”content://net.dxs.provider/person/* ” *来匹配任意文本
要给第三方提供数据访问的ContentProvider类
DxsProvider.java
package net.dxs.sqlite.provider; import net.dxs.sqlite.dao.MyHelper; import android.content.ContentProvider; import android.content.ContentUris; import android.content.ContentValues; import android.content.UriMatcher; import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import android.net.Uri; public class DxsProvider extends ContentProvider { private MyHelper helper; private UriMatcher matcher = new UriMatcher(UriMatcher.NO_MATCH); // 创建匹配器对象, 当Uri无法匹配的时候, 得到-1 private static final int ACCOUNT_ID = 0; private static final int ACCOUNT = 1; private static final int ORDER = 2; @Override public boolean onCreate() { // 创建之后执行 System.out.println("DxsProvider--->onCreate"); helper = new MyHelper(getContext()); matcher.addURI("net.dxs.provider", "account/#", ACCOUNT_ID); matcher.addURI("net.dxs.provider", "account", ACCOUNT); // 向匹配器中添加可以识别的Uri, 指定结果码 matcher.addURI("net.dxs.provider", "order", ORDER); return false; } @Override public Uri insert(Uri uri, ContentValues values) { System.out.println("DxsProvider--->insert"); switch (matcher.match(uri)) { // 使用匹配器识别Uri, 得到预先指定的结果码 case ACCOUNT: SQLiteDatabase db = helper.getWritableDatabase(); long id = db.insert("account", "_id", values); // 通过values拼接SQL语句 getContext().getContentResolver().notifyChange(uri, null); db.close(); return ContentUris.withAppendedId(uri, id); // 把新记录的id拼在Uri最后, 返回 case ORDER: System.out.println("暂时没有order表"); return null; default: throw new IllegalArgumentException("Uri无法识别: " + uri); } } @Override public int delete(Uri uri, String selection, String[] selectionArgs) { System.out.println("DxsProvider--->delete"); switch (matcher.match(uri)) { case ACCOUNT_ID: long id = ContentUris.parseId(uri); selection = "_id=?"; selectionArgs = new String[] { id + "" }; case ACCOUNT: SQLiteDatabase db = helper.getWritableDatabase(); int count = db.delete("account", selection, selectionArgs); getContext().getContentResolver().notifyChange(uri, null); db.close(); return count; default: throw new IllegalArgumentException("Uri无法识别: " + uri); } } @Override public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) { System.out.println("DxsProvider--->update"); switch (matcher.match(uri)) { case ACCOUNT_ID: long id = ContentUris.parseId(uri); selection = "_id=?"; selectionArgs = new String[] { id + "" }; case ACCOUNT: SQLiteDatabase db = helper.getWritableDatabase(); int count = db.update("account", values, selection, selectionArgs); getContext().getContentResolver().notifyChange(uri, null); // 当数据改变时, 发送通知, 这时ContentObserver就会接收到 db.close(); return count; default: throw new IllegalArgumentException("Uri无法识别: " + uri); } } @Override public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) { System.out.println("DxsProvider--->query"); switch (matcher.match(uri)) { case ACCOUNT_ID: long id = ContentUris.parseId(uri); // 截取最后一个"/"后面的数字, 转为long selection = "_id=?"; selectionArgs = new String[] { id + "" }; case ACCOUNT: SQLiteDatabase db = helper.getReadableDatabase(); Cursor c = db.query("account", projection, selection, selectionArgs, null, null, sortOrder); return c; default: throw new IllegalArgumentException("Uri无法识别: " + uri); } } @Override public String getType(Uri uri) { // 获取Uri的MimeType, text/html text/css image/jpg audio/mp3 System.out.println("DxsProvider--->getType"); switch (matcher.match(uri)) { case ACCOUNT_ID: return "vnd.android.cursor.item/account"; case ACCOUNT: return "vnd.android.cursor.dir/account"; default: throw new IllegalArgumentException("Uri无法识别: " + uri); } } }
注意清单文件要注册声明provider
<provider android:name="net.dxs.sqlite.provider.DxsProvider" android:authorities="net.dxs.provider" android:exported="true" />
第三方APP开始调用提供的ContentProvider
package net.dxs.other; import android.content.ContentResolver; import android.content.ContentValues; import android.database.Cursor; import android.net.Uri; import android.test.AndroidTestCase; public class ProviderTest extends AndroidTestCase { public void test1() { ContentResolver resolver = getContext().getContentResolver(); // 获取ContentResolver Uri uri = Uri.parse("content://net.dxs.provider"); // 指定ContentProvider的Uri // resolver.delete(uri, null, null); // 对指定Uri调用删除方法 resolver.query(uri, null, null, null, null); } public void testInsert() { ContentResolver resolver = getContext().getContentResolver(); Uri uri = Uri.parse("content://net.dxs.provider/account"); ContentValues values = new ContentValues(); values.put("name", "insert"); values.put("balance", 23456); System.out.println(resolver.insert(uri, values)); // 得到刚刚插入的Uri } public void testDelete() { ContentResolver resolver = getContext().getContentResolver(); Uri uri = Uri.parse("content://net.dxs.provider/account/20"); resolver.delete(uri, null, null); } public void testUpdate() { ContentResolver resolver = getContext().getContentResolver(); Uri uri = Uri.parse("content://net.dxs.provider/account/1"); ContentValues values = new ContentValues(); values.put("name", "深情小建"); values.put("balance", 20000); resolver.update(uri, values, null, null); } public void testQuery() { ContentResolver resolver = getContext().getContentResolver(); Uri uri = Uri.parse("content://net.dxs.provider/account/100"); Cursor c = resolver.query(uri, null, null, null, null); while (c.moveToNext()) { System.out.println(c.getString(c.getColumnIndex("name")) + ": " + c.getInt(c.getColumnIndex("balance"))); } c.close(); } public void testGetType() { ContentResolver resolver = getContext().getContentResolver(); System.out.println(resolver.getType(Uri.parse("content://net.dxs.provider/account/100"))); // 单条记录, 返回item System.out.println(resolver.getType(Uri.parse("content://net.dxs.provider/account"))); // 多条记录, 返回dir } }
生成的数据库表如图
实例源代码->百度网盘