AndroidAZ系列:ContentProvider(All,Face)

Git倉(cāng)庫地址

配套四維導(dǎo)圖地址

AndroidAZ系列有以下目的:

  1. Android程序猿的面試(初級(jí),中級(jí),高級(jí),資深),拿到滿意的offer。
  2. Android程序猿學(xué)習(xí)進(jìn)階。

標(biāo)記說明:因?yàn)楣P者是列出所有的Android知識(shí)點(diǎn),因此面試不需要看那么多內(nèi)容,如果是面試的知識(shí)點(diǎn)。筆者會(huì)加上標(biāo)記Face,而如果不是面試的知識(shí)點(diǎn),筆者會(huì)加上No標(biāo)記,它是要學(xué)的東西;然后筆者將Android面試者或者面試者分為4個(gè)等級(jí),初級(jí)A1,中級(jí)A2,高級(jí)A3,資深A(yù)4,如果這個(gè)知識(shí)點(diǎn)是所有等級(jí)的范圍,那么筆者將會(huì)以all標(biāo)記上。因此進(jìn)階路線就是A1->A2->A3->A4。也是面試者挑選的復(fù)習(xí)范圍,假如你是中級(jí)程序員,那么你面試要看的內(nèi)容就是包含A2&Face的標(biāo)記。

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

  • A1: 初級(jí)Android工程師。

  • A2: 中級(jí)Android工程師。

  • A3: 高級(jí)Android工程師。

  • A4: 資深A(yù)ndroid工程師。

  • Face: 是面試的知識(shí)點(diǎn)。

  • No: 面試基本遇不到。

1.ContentProvider是什么

ContentProvider一般為存儲(chǔ)和獲取數(shù)據(jù)提供統(tǒng)一的接口,可以在不同的應(yīng)用程序之間共享數(shù)據(jù).

  • 1,ContentProvider提供了對(duì)底層數(shù)據(jù)存儲(chǔ)方式的抽象。比如下圖中,底層使用了SQLite數(shù)據(jù)庫,在用了ContentProvider封裝后,即使你把數(shù)據(jù)庫換成MongoDB,也不會(huì)對(duì)上層數(shù)據(jù)使用層代碼產(chǎn)生影響。
image.png
  • Android框架中的一些類需要ContentProvider類型數(shù)據(jù)。如果你想讓你的數(shù)據(jù)可以使用在如SyncAdapter, Loader, CursorAdapter等類上,那么你就需要為你的數(shù)據(jù)做一層ContentProvider封裝。

  • 第三個(gè)原因也是最主要的原因,是ContentProvider為應(yīng)用間的數(shù)據(jù)交互提供了一個(gè)安全的環(huán)境。它準(zhǔn)許你把自己的應(yīng)用數(shù)據(jù)根據(jù)需求開放給其他應(yīng)用進(jìn)行增、刪、改、查,而不用擔(dān)心直接開放數(shù)據(jù)庫權(quán)限而帶來的安全問題。

2.ContentProvider的使用

Content Provider的用法有兩種,一種是使用現(xiàn)有的content provider來讀取和操作相應(yīng)程序中的數(shù)據(jù),另一種是創(chuàng)建自己的content provider提供外部訪問接口.
要像訪問content provider中共享的數(shù)據(jù),要借助content resolver類,content resolver提供了一系列的方法對(duì)數(shù)據(jù)進(jìn)行增刪改查操作,以URI為參數(shù).

2.1 URI

  • 定義:Uniform Resource Identifier,即統(tǒng)一資源標(biāo)識(shí)符

  • 作用: 這里用來標(biāo)識(shí) ContentProvider 和其中的數(shù)據(jù),外界進(jìn)程通過 URI 找到對(duì)應(yīng)的ContentProvider 和其中的數(shù)據(jù),再進(jìn)行數(shù)據(jù)操作

  • 具體使用:URI分為 系統(tǒng)預(yù)置 & 自定義,分別對(duì)應(yīng)系統(tǒng)內(nèi)置的數(shù)據(jù)(如通訊錄、日程表等等)和自定義數(shù)據(jù)庫

image.png
``` java
// 設(shè)置URI
Uri uri = Uri.parse("content://com.carson.provider/User/1")
// 上述URI指向的資源是:名為 `com.carson.provider`的`ContentProvider` 中表名 為`User` 中的 `id`為1的數(shù)據(jù)

// 特別注意:URI模式存在匹配通配符* & #

// *:匹配任意長(zhǎng)度的任何有效字符的字符串
// 以下的URI 表示 匹配provider的任何內(nèi)容
content://com.example.app.provider/*
// #:匹配任意長(zhǎng)度的數(shù)字字符的字符串
// 以下的URI 表示 匹配provider中的table表的所有行
content://com.example.app.provider/table/#

```

2.2 ContentResolver使用方法

``` java
// 使用ContentResolver前,需要先獲取ContentResolver
// 可通過在所有繼承Context的類中 通過調(diào)用getContentResolver()來獲得ContentResolver
ContentResolver resolver =  getContentResolver();

// 設(shè)置ContentProvider的URI
Uri uri = Uri.parse("content://cn.scu.myprovider/user");

// 根據(jù)URI 操作 ContentProvider中的數(shù)據(jù)
// 此處是獲取ContentProvider中 user表的所有記錄
Cursor cursor = resolver.query(uri, null, null, null, "userid desc");
```

3.如何創(chuàng)建自定義ContentProvider

首先我們創(chuàng)建一個(gè)自己的TestProvider繼承ContentProvider。默認(rèn)該P(yáng)rovider需要實(shí)現(xiàn)如下六個(gè)方法,

onCreate(),

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

insert(Uri, ContentValues),

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

delete(Uri, String, String[]),

getType(Uri)

方法的具體介紹可以參考 官網(wǎng)

下面我們以實(shí)現(xiàn)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;
}

此例中我們可以看到,我們根據(jù)path的不同,來區(qū)別對(duì)不同的數(shù)據(jù)庫表進(jìn)行操作,從而完成uri與具體數(shù)據(jù)庫間的映射關(guān)系。

因?yàn)镃ontentProvider作為四大組件之一,所以還需要在AndroidManifest.xml中注冊(cè)一下。

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

4.ContentProvider原理

使用binder在不同程序間傳遞數(shù)據(jù).

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容