AndroidAZ系列:ContentProvider(All,

作者: 道阻且长_行则将至 | 来源:发表于2019-11-08 20:08 被阅读0次

    Git仓库地址

    配套四维导图地址

    AndroidAZ系列有以下目的:

    1. Android程序猿的面试(初级,中级,高级,资深),拿到满意的offer。
    2. Android程序猿学习进阶。

    标记说明:因为笔者是列出所有的Android知识点,因此面试不需要看那么多内容,如果是面试的知识点。笔者会加上标记Face,而如果不是面试的知识点,笔者会加上No标记,它是要学的东西;然后笔者将Android面试者或者面试者分为4个等级,初级A1,中级A2,高级A3,资深A4,如果这个知识点是所有等级的范围,那么笔者将会以all标记上。因此进阶路线就是A1->A2->A3->A4。也是面试者挑选的复习范围,假如你是中级程序员,那么你面试要看的内容就是包含A2&Face的标记。

    • All : 所有的Android工程师都看。

    • A1: 初级Android工程师。

    • A2: 中级Android工程师。

    • A3: 高级Android工程师。

    • A4: 资深Android工程师。

    • Face: 是面试的知识点。

    • No: 面试基本遇不到。

    1.ContentProvider是什么

    ContentProvider一般为存储和获取数据提供统一的接口,可以在不同的应用程序之间共享数据.

    • 1,ContentProvider提供了对底层数据存储方式的抽象。比如下图中,底层使用了SQLite数据库,在用了ContentProvider封装后,即使你把数据库换成MongoDB,也不会对上层数据使用层代码产生影响。
    image.png
    • Android框架中的一些类需要ContentProvider类型数据。如果你想让你的数据可以使用在如SyncAdapter, Loader, CursorAdapter等类上,那么你就需要为你的数据做一层ContentProvider封装。

    • 第三个原因也是最主要的原因,是ContentProvider为应用间的数据交互提供了一个安全的环境。它准许你把自己的应用数据根据需求开放给其他应用进行增、删、改、查,而不用担心直接开放数据库权限而带来的安全问题。

    2.ContentProvider的使用

    Content Provider的用法有两种,一种是使用现有的content provider来读取和操作相应程序中的数据,另一种是创建自己的content provider提供外部访问接口.
    要像访问content provider中共享的数据,要借助content resolver类,content resolver提供了一系列的方法对数据进行增删改查操作,以URI为参数.

    2.1 URI

    • 定义:Uniform Resource Identifier,即统一资源标识符

    • 作用: 这里用来标识 ContentProvider 和其中的数据,外界进程通过 URI 找到对应的ContentProvider 和其中的数据,再进行数据操作

    • 具体使用:URI分为 系统预置 & 自定义,分别对应系统内置的数据(如通讯录、日程表等等)和自定义数据库

    image.png
    ``` java
    // 设置URI
    Uri uri = Uri.parse("content://com.carson.provider/User/1")
    // 上述URI指向的资源是:名为 `com.carson.provider`的`ContentProvider` 中表名 为`User` 中的 `id`为1的数据
    
    // 特别注意:URI模式存在匹配通配符* & #
    
    // *:匹配任意长度的任何有效字符的字符串
    // 以下的URI 表示 匹配provider的任何内容
    content://com.example.app.provider/*
    // #:匹配任意长度的数字字符的字符串
    // 以下的URI 表示 匹配provider中的table表的所有行
    content://com.example.app.provider/table/#
    
    ```
    

    2.2 ContentResolver使用方法

    ``` java
    // 使用ContentResolver前,需要先获取ContentResolver
    // 可通过在所有继承Context的类中 通过调用getContentResolver()来获得ContentResolver
    ContentResolver resolver =  getContentResolver();
    
    // 设置ContentProvider的URI
    Uri uri = Uri.parse("content://cn.scu.myprovider/user");
    
    // 根据URI 操作 ContentProvider中的数据
    // 此处是获取ContentProvider中 user表的所有记录
    Cursor cursor = resolver.query(uri, null, null, null, "userid desc");
    ```
    

    3.如何创建自定义ContentProvider

    首先我们创建一个自己的TestProvider继承ContentProvider。默认该Provider需要实现如下六个方法,

    onCreate(),

    query(Uri, String[], String, String[], String),

    insert(Uri, ContentValues),

    update(Uri, ContentValues, String, String[]),

    delete(Uri, String, String[]),

    getType(Uri)

    方法的具体介绍可以参考 官网

    下面我们以实现insert和query方法为例

    private final static int TEST = 100;
    
    static UriMatcher buildUriMatcher() {
        final UriMatcher matcher = new UriMatcher(UriMatcher.NO_MATCH);
        final String authority = TestContract.CONTENT_AUTHORITY;
    
        matcher.addURI(authority, TestContract.PATH_TEST, TEST);
    
        return matcher;
    }
    
    @Nullable
    @Override
    public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {
        final SQLiteDatabase db = mOpenHelper.getReadableDatabase();
    
        Cursor cursor = null;
        switch ( buildUriMatcher().match(uri)) {
            case TEST:
                cursor = db.query(TestContract.TestEntry.TABLE_NAME, projection, selection, selectionArgs, sortOrder, null, null);
                break;
        }
    
        return cursor;
    }
    
    @Nullable
    @Override
    public Uri insert(Uri uri, ContentValues values) {
        final SQLiteDatabase db = mOpenHelper.getWritableDatabase();
        Uri returnUri;
        long _id;
        switch ( buildUriMatcher().match(uri)) {
            case TEST:
                _id = db.insert(TestContract.TestEntry.TABLE_NAME, null, values);
                if ( _id > 0 )
                    returnUri = TestContract.TestEntry.buildUri(_id);
                else
                    throw new android.database.SQLException("Failed to insert row into " + uri);
                break;
            default:
                throw new android.database.SQLException("Unknown uri: " + uri);
        }
        return returnUri;
    }
    

    此例中我们可以看到,我们根据path的不同,来区别对不同的数据库表进行操作,从而完成uri与具体数据库间的映射关系。

    因为ContentProvider作为四大组件之一,所以还需要在AndroidManifest.xml中注册一下。

    <provider
        android:authorities="me.pengtao.contentprovidertest"
        android:name=".provider.TestProvider"
    />
    

    4.ContentProvider原理

    使用binder在不同程序间传递数据.

    相关文章

      网友评论

        本文标题:AndroidAZ系列:ContentProvider(All,

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