ContentProvider

ContentProvider 在android中的作用是对外共享数据,也就是说你可以通过ContentProvider把应用中的数据共享给其他应用访问,其他应用可以通过ContentProvider 对你应用中的数据进行添删改查。
ContentProvider的就是自定义增删改查接口并暴露出去,让别的应用访问自己的数据。ContentResolver就是按照一定规则访问内容提供者的数据。

ContentProvider对外共享数据步骤:

步骤

1. 定义一个类 继承 ContentProvider
2. 定义匹配规则 uri
3. 通过静态代码块添加匹配规则 
4.在manifest.xml中配置contentProvider.

Uri介绍

uri代表了要操作的数据

上面我们提到了Android提供内容的叫Provider,那么在Android中怎么区分各个Provider?

Uri作为唯一的标识来标识这个Provider。

ContentProvider的scheme为:content://
Authority 用于唯一标识这个ContentProvider,外部调用者可以根据这个标识来找到它。
路径(path)可以用来表示我们要操作的数据,路径的构建应根据业务而定,如下:
要操作file表中id为10的记录,可以构建这样的路径:/file/10
要操作file表中id为10的记录的name字段, file/10/name
要操作file表中的所有记录,可以构建这样的路径:/file

当然要操作的数据可以是数据库,也可以是文件、xml或网络等其他存储方式。

代码示例

public class FileProvider  extends ContentProvider {

    private Context mContext;
private static final int QUEYSUCESS = 0;
private static final int INSERTSUCESS = 1;
//UriMatcher.NO_MATCH表示不匹配任何路径的返回码
private static UriMatcher mUriMatcher = new UriMatcher(UriMatcher.NO_MATCH); private SQLiteDatabase mDb;
private String mTableName = DbOpenHelper.STUDENT_TABLE_NAME;
static{
//注册所有要匹配的uri
mUriMatcher.addURI("com.itcast.contentp.FileProvider", "query", QUEYSUCESS);
mUriMatcher.addURI("com.itcast.contentp.FileProvider", "insert", INSERTSUCESS);
} //该方法在其它应用第一次访问它时才会被创建
@Override
public boolean onCreate() {
mContext = getContext();
mDb = new DbOpenHelper(mContext).getWritableDatabase();
return false;
} /**
*public final Cursor query (Uri uri, String[] projection,String selection,String[] selectionArgs, String sortOrder)
*projection : 这个参数告诉查询要返回的列(Column)即需要的字段,比如Contacts Provider提供了联系人的ID和联系人的NAME等内容.
*selection :查询where字句
*selectionArgs : 查询条件属性值
*sortOrder :结果排序
*/
@Override
public Cursor query(Uri uri, String[] arg1, String arg2, String[] arg3,String arg4) {
if (mUriMatcher.match(uri)== QUEYSUCESS ) {//uri匹配后进行下面的操作
Cursor cursor = mDb.query(tableName, arg1, arg2, arg3, null, null, null);
getContext().getContentResolver().notifyChange(uri, null);
return cursor;
}else{
throw new IllegalArgumentException("match fail");
}
}
这里只给出部分代码。。。。。。

ContentResolver

使用ContentResolver调用ContentProvider提供的接口,对ContentProvider中的数据进行添加、删除、修改和查询操作时。

可以使用Activity提供的getContentResolver()方法来获取ContentResolver对象。 ContentResolver 类提供了与ContentProvider类相同签名的四个方法:

  • public Uri insert(Uri uri, ContentValues values)
  • public int delete(Uri uri, String selection, String[] selectionArgs)。
  • public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs)
  • public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder)

1.对ContentProvider中的数据进行增删改查

直接看代码:

//uriQuery必须与要查询的ContentProvider中的要操作数据的uri保持一致(btw 这里只给了查询好插入的例子)

ContentValues values = new ContentValues();
Cursor cursor = getContentResolver().query(uriQuery, null, null, null, null);
int count = cursor.getCount(); //获取到一共有多少行
int contact_id = count + 1; ContentValues nameValues = new ContentValues();
nameValues.put("name", name);
nameValues.put("mime_type", "vnd.android.cursor.item/name");
nameValues.put("contact_id", contact_id);
getContentResolver().insert(uriInsert, nameValues);

2.监听ContentProvider中数据的变化

在ContentProvider 发生数据变化时调用getContentResolver().notifyChange(uri, null)来通知注册在此URI上的访问者。

当数据放生变化时会调用ContentObserver的onChange()来进行一系列的后续操作~~~

如下:

public class MainActivity extends Activity {

    private Uri uri;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); //注册
uri = Uri.parse("content://com.example.contentp.AccountProvider");
        getContentResolver().registerContentObserver(uri, true, new MyObserver(new Handler())) 
}
}

//监听到变化后调用onChange()来执行一系列操作

private class MyObserver extends ContentObserver {
Uri uri = Uri.parse("content://com.example.contentp.AccountProvider"); public MyObserver(Handler handler) { super(handler);
}
@Override
public void onChange(boolean selfChange) {
Cursor cursor = getContentResolver().query(uri, new String[]{"file","mime_type","date"}, null, null, null);
while(cursor.moveToNext()){
//执行一些操作
}
}
}

ContentProviderClient

与ContentResolver一样都是用来对ContentProvider中的数据进行添加、删除、修改和查询操作的

通过调用 getContentResolver().acquireContentProviderClient(authority) 方法获取 ContentProviderClient对象。

用法跟ContentResolver相似,不同点是ContentProviderClient 对象必须在结束使用后,调用ContentProviderClient.release()来释放。这会是系统释放对应的ContentProvider对象。

对于相同ContentProvider 的多次调用,推荐使用ContentProviderClient。

ContentProvider和ContentResolver的使用的更多相关文章

  1. ContentProvider、ContentResolver、ContentObserver之间的关系

    ContentProvider.ContentResolver.ContentObserver之间的关系 ContentPRrovider: * 四大组件的内容提供者,主要用于对外提供数据 * 实现各 ...

  2. Android ContentProvider、ContentResolver和ContentObserver的使用

    1.ContentProvider.ContentResolver和ContentObserver ContentProvider是Android的四大组件之中的一个,可见它在Android中的作用非 ...

  3. ContentProvider 、 ContentResolver 、 ContentObserver

    说说ContentProvider . ContentResolver . ContentObserver 之间的关系**a. ContentProvider 内容提供者,用于对外提供数据 b. Co ...

  4. android 多应用程序数据共享 ContentProvider和ContentResolver

      android 没有一个可以将所有应用程序数据统一放置的地方,即两个应用程序间的数据不能共享.但ContentProvider与ContentResolver可以解决多应用程序数据共享. 我们都知 ...

  5. ContentProvider,ContentResolver和ContentObserver 使用

    1 ContentProvider内容提供者 四大组件之一,实现不同程序实现数据的共享.联系人应用就使用了ContentProvider,比如你在自己的应用可以读取和修改联系人的数据(获得相应权限). ...

  6. ContentProvider与ContentResolver使用【转】

    这篇文章被转载而转载者未注明原文出处,在此未加上原文地址链接,本人向原作者致以歉意. 下面是文章内容: 使用ContentProvider共享数据: 当应用继承ContentProvider类,并重写 ...

  7. ContentProvider与ContentResolver使用

    例如以下内容为从网络转载: 使用ContentProvider共享数据: 当应用继承ContentProvider类,并重写该类用于提供数据和存储数据的方法,就能够向其它应用共享其数据.虽然使用其它方 ...

  8. ContentProvider与ContentResolver

    使用ContentProvider共享数据: 当应用继承ContentProvider类,并重写该类用于提供数据和存储数据的方法,就可以向其他应用共享其数据.虽然使用其他方法也可以对外共享数据,但数据 ...

  9. ContentProvider官方教程(2)简介、Content URIs

    In this document Overview Accessing a provider Content URIs Content Provider Basics A content provid ...

随机推荐

  1. DX11 Without DirectX SDK--05 键盘和鼠标输入

    回到 DirectX11--使用Windows SDK来进行开发 提供键鼠输入可以说是一个游戏的必备要素.在这里,我们不使用DirectInput,因为Windws SDK本身就不提供该头文件.这里我 ...

  2. redis的持久化之AOF

    AOF Redis 分别提供了 RDB 和 AOF 两种持久化机制: RDB 将数据库的快照(snapshot)以二进制的方式保存到磁盘中. AOF 则以协议文本的方式,将所有对数据库进行过写入的命令 ...

  3. 关于java多线程关键字volatile的理解

    volatile关键字的作用是强制从公共堆栈中取得变量的值,而不是从线程私有数据栈中取得变量的值. 使用volition关键字增加了实例变量在多个线程间的可见性.但volition有个致命的缺点就是不 ...

  4. Java NIO核心组件简介

    原文链接:http://tutorials.jenkov.com/java-nio/overview.html NIO包含下面几个核心的组件: Channels Buffer Selector 整个N ...

  5. Oracle-08:连接查询

    ------------吾亦无他,唯手熟尔,谦卑若愚,好学若饥------------- 首先提供数据库脚本,供测试使用 create table DEPT ( deptno ) not null, ...

  6. Linux kernel的中断子系统之(六):ARM中断处理过程

    返回目录:<ARM-Linux中断系统>. 总结:二中断处理经过两种模式:IRQ模式和SVC模式,这两种模式都有自己的stack,同时涉及到异常向量表中的中断向量. 三ARM处理器在感知到 ...

  7. 【手记】解决启动SQL Server Management Studio 17时报Cannot find one of more components...的问题

    刚装好SSMS 17.1准备体验,弹出: 一番搜索,普遍办法都是安装VS2015独立shell.删除某个注册表项什么的,没用,首先这个shell我是装了的,然后也没有那个注册表项.我自己尝试过重装sh ...

  8. spring boot -Properties & configuration

    72. Properties & configuration72.1 Automatically expand properties at build timeRather than hard ...

  9. 前端学习总结(一)——常见数据结构的javascript实现

    1.列表类 // 列表类 function List() { this.listSize = 0; // 列表的元素个数 this.pos = 0; // 列表的当前位置 this.dataStore ...

  10. guid.go

    ] = byte(g)     hex.Encode(h[:], b[:])     return h }