美文网首页
ContentProvider和ContentResolver的

ContentProvider和ContentResolver的

作者: 简书_大叔 | 来源:发表于2019-10-23 16:03 被阅读0次

ContentProvider

ContentProvider 在Android中的作用是对外共享数据,也就是说可以通过ContentProvider把应用中的数据共享给其他应用访问,其他应用可以通过ContentProvider对你应用中的数据进行增删改查。

ContentProvider就是自定义增删改查接口并暴露出去,让别的应用访问自己的数据。ContentProvider就是按照一定规则访问内容提供者的数据。

ContentProvider对外共享数据步骤:

1.定义一个类 继承 ContentProvider

2.定义匹配规则 Uri

3.通过静态代码块添加匹配规则

4.在manifest.xml中配置contentProvider

Uri介绍

uri代表了要操作的数据

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

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

ContentProvider的scheme为:content://

Authority用于唯一标识这个ContenProvider,外部调用者可以根据这个标识来找到它。

路径(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");

        }

}

这里只给出部分代码。。。。。。

ContenResolver

使用ContentResoler调用ContentProvider提供的接口,对ContentProvider中的数据进行添加、删除、修改和查询操作时,可以使用Activity提供的getContentResolver()方法来获取ContentResolver对象。ContentResolver类提供了与ContentProvider类相同签名的四个方法:

public Uri insert(Uri uri,ContentValues values)

public int delete(Uri uri,String selection,String[] seletionArgs)

public int update(Uri uri,ContentValues values,String seletion,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中的数据进行添加、删除、修改和查询操作的

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

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

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

相关文章

网友评论

      本文标题:ContentProvider和ContentResolver的

      本文链接:https://www.haomeiwen.com/subject/cgrkvctx.html