AndroidAZ系列有以下目的:
- Android程序猿的面試(初級,中級,高級,資深),拿到滿意的offer。
- Android程序猿學(xué)習(xí)進(jìn)階。
標(biāo)記說明:因?yàn)楣P者是列出所有的Android知識點(diǎn),因此面試不需要看那么多內(nèi)容,如果是面試的知識點(diǎn)。筆者會加上標(biāo)記Face,而如果不是面試的知識點(diǎn),筆者會加上No標(biāo)記,它是要學(xué)的東西;然后筆者將Android面試者或者面試者分為4個等級,初級A1,中級A2,高級A3,資深A(yù)4,如果這個知識點(diǎn)是所有等級的范圍,那么筆者將會以all標(biāo)記上。因此進(jìn)階路線就是A1->A2->A3->A4。也是面試者挑選的復(fù)習(xí)范圍,假如你是中級程序員,那么你面試要看的內(nèi)容就是包含A2&Face的標(biāo)記。
All : 所有的Android工程師都看。
A1: 初級Android工程師。
A2: 中級Android工程師。
A3: 高級Android工程師。
A4: 資深A(yù)ndroid工程師。
Face: 是面試的知識點(diǎn)。
No: 面試基本遇不到。
1.ContentProvider是什么
ContentProvider一般為存儲和獲取數(shù)據(jù)提供統(tǒng)一的接口,可以在不同的應(yīng)用程序之間共享數(shù)據(jù).
- 1,ContentProvider提供了對底層數(shù)據(jù)存儲方式的抽象。比如下圖中,底層使用了SQLite數(shù)據(jù)庫,在用了ContentProvider封裝后,即使你把數(shù)據(jù)庫換成MongoDB,也不會對上層數(shù)據(jù)使用層代碼產(chǎn)生影響。
Android框架中的一些類需要ContentProvider類型數(shù)據(jù)。如果你想讓你的數(shù)據(jù)可以使用在如SyncAdapter, Loader, CursorAdapter等類上,那么你就需要為你的數(shù)據(jù)做一層ContentProvider封裝。
第三個原因也是最主要的原因,是ContentProvider為應(yīng)用間的數(shù)據(jù)交互提供了一個安全的環(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提供了一系列的方法對數(shù)據(jù)進(jìn)行增刪改查操作,以URI為參數(shù).
2.1 URI
定義:Uniform Resource Identifier,即統(tǒng)一資源標(biāo)識符
作用: 這里用來標(biāo)識 ContentProvider 和其中的數(shù)據(jù),外界進(jìn)程通過 URI 找到對應(yīng)的ContentProvider 和其中的數(shù)據(jù),再進(jìn)行數(shù)據(jù)操作
具體使用:URI分為 系統(tǒng)預(yù)置 & 自定義,分別對應(yīng)系統(tǒng)內(nèi)置的數(shù)據(jù)(如通訊錄、日程表等等)和自定義數(shù)據(jù)庫
``` java
// 設(shè)置URI
Uri uri = Uri.parse("content://com.carson.provider/User/1")
// 上述URI指向的資源是:名為 `com.carson.provider`的`ContentProvider` 中表名 為`User` 中的 `id`為1的數(shù)據(jù)
// 特別注意:URI模式存在匹配通配符* & #
// *:匹配任意長度的任何有效字符的字符串
// 以下的URI 表示 匹配provider的任何內(nèi)容
content://com.example.app.provider/*
// #:匹配任意長度的數(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)建一個自己的TestProvider繼承ContentProvider。默認(rèn)該P(yáng)rovider需要實(shí)現(xiàn)如下六個方法,
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ū)別對不同的數(shù)據(jù)庫表進(jìn)行操作,從而完成uri與具體數(shù)據(jù)庫間的映射關(guān)系。
因?yàn)镃ontentProvider作為四大組件之一,所以還需要在AndroidManifest.xml中注冊一下。
<provider
android:authorities="me.pengtao.contentprovidertest"
android:name=".provider.TestProvider"
/>
4.ContentProvider原理
使用binder在不同程序間傳遞數(shù)據(jù).