Android四大組件之內(nèi)容提供者

ContentProvider
是Android中提供的專門用于不同應(yīng)用之間進(jìn)行數(shù)據(jù)共享的方式
底層由Binder機(jī)制實(shí)現(xiàn)茫蛹,由于系統(tǒng)對他做了封裝手报,使用起來比AIDL方便許多黄锤,無需關(guān)注底層細(xì)節(jié)即可輕松進(jìn)行IPC
Provider的onCreate由Server端主線程調(diào)用骗爆,所以不要做耗時(shí)任務(wù)
CRUD操作由Binder線程調(diào)用,所以可以做并發(fā)操作度陆,不會(huì)引起堵塞

Server


// 服務(wù)端數(shù)據(jù)庫
public class DbOpenHelper extends SQLiteOpenHelper {
    private static final String DB_NAME = "providers.db";
    private static final int DB_VERSION = 1;

    public static final String BOOK_TABLE_NAME = "book";
    public static final String USER_TABLE_NAME = "user";

    private static final String CREATE_BOOK_TABLE = "CREATE TABLE IF NOT EXISTS " + BOOK_TABLE_NAME + "(" +
            "_id    INTEGER PRIMARY KEY, " +
            "name   TEXT" +
            ")";

    private static final String CREATE_USER_TABLE = "CREATE TABLE IF NOT EXISTS " + USER_TABLE_NAME + "(" +
            "_id    INTEGER PRIMARY KEY, " +
            "name   TEXT," +
            "sex    INTEGER" +
            ")";


    public DbOpenHelper(Context context) {
        super(context, DB_NAME, null, DB_VERSION);
    }


    @Override
    public void onCreate(SQLiteDatabase db) {
        db.execSQL(CREATE_BOOK_TABLE);
        db.execSQL(CREATE_USER_TABLE);
    }

    @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
        // TODO: 2016/7/29 ignore
    }
}
// 實(shí)現(xiàn)內(nèi)容提供者
public class CustomProvider extends ContentProvider {
    public static final String TAG = "CustomProvider";
    public static final String AUTHORITY = "com.rtfsc.auth.provider";

    public static final Uri BOOK_CONTENT_URI = Uri.parse("content://"
            + AUTHORITY + "/book");
    public static final Uri USER_CONTENT_URI = Uri.parse("content://"
            + AUTHORITY + "/user");

    public static final int BOOK_URI_CODE = 0;
    public static final int USER_URI_CODE = 1;

    private static final UriMatcher sUriMatcher = new UriMatcher(
            UriMatcher.NO_MATCH);

    private Context mContext;
    private SQLiteDatabase mDatabase;

    static {
        sUriMatcher.addURI(AUTHORITY, "book", BOOK_URI_CODE);
        sUriMatcher.addURI(AUTHORITY, "user", USER_URI_CODE);
    }

    private String getTableName(Uri uri) {
        String tableName = null;
        switch (sUriMatcher.match(uri)) {
            case BOOK_URI_CODE:
                tableName = DbOpenHelper.BOOK_TABLE_NAME;
                break;
            case USER_URI_CODE:
                tableName = DbOpenHelper.USER_TABLE_NAME;
                break;
            default:
                break;
        }
        return tableName;
    }

    @Override
    public boolean onCreate() {
        Log.d(TAG, "onCreate, current thread: " + Thread.currentThread().getName());
        mContext = getContext();
        mDatabase = new DbOpenHelper(mContext).getWritableDatabase();
        initData();
        return true;
    }

    private void initData() {
        mDatabase.execSQL("delete from " + DbOpenHelper.BOOK_TABLE_NAME);
        mDatabase.execSQL("delete from " + DbOpenHelper.USER_TABLE_NAME);

        mDatabase.execSQL("insert into " + DbOpenHelper.BOOK_TABLE_NAME + " values(1,'Android');");
        mDatabase.execSQL("insert into " + DbOpenHelper.BOOK_TABLE_NAME + " values(2,'iOS');");
        mDatabase.execSQL("insert into " + DbOpenHelper.BOOK_TABLE_NAME + " values(3,'Pytnon');");

        mDatabase.execSQL("insert into " + DbOpenHelper.USER_TABLE_NAME + " values(1,'Wong',1);");
        mDatabase.execSQL("insert into " + DbOpenHelper.USER_TABLE_NAME + " values(2,'Kim',1);");
        mDatabase.execSQL("insert into " + DbOpenHelper.USER_TABLE_NAME + " values(3,'Hung',1);");
    }

    @Nullable
    @Override
    public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {
        Log.d(TAG, "query, current thread: " + Thread.currentThread().getName());
        String tableName = getTableName(uri);
        if (tableName.isEmpty()) {
            throw new IllegalArgumentException("Unsupport URI: " + uri);
        }
        return mDatabase.query(tableName, projection, selection, selectionArgs, null, null, null);
    }

    @Nullable
    @Override
    public String getType(Uri uri) {
        Log.d(TAG, "getType");
        return null;
    }

    @Nullable
    @Override
    public Uri insert(Uri uri, ContentValues values) {
        Log.d(TAG, "insert");
        String tableName = getTableName(uri);
        if (tableName.isEmpty()) {
            throw new IllegalArgumentException("Unsupport URI: " + uri);
        }
        mDatabase.insert(tableName, null, values);
        mContext.getContentResolver().notifyChange(uri, null);
        return uri;
    }

    @Override
    public int delete(Uri uri, String selection, String[] selectionArgs) {
        Log.d(TAG, "delete");
        String tableName = getTableName(uri);
        if (tableName.isEmpty()) {
            throw new IllegalArgumentException("Unsupport URI: " + uri);
        }
        int count = mDatabase.delete(tableName, selection, selectionArgs);
        if (count > 0) {
            mContext.getContentResolver().notifyChange(uri, null);
        }
        return count;
    }

    @Override
    public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {
        Log.d(TAG, "update");
        String tableName = getTableName(uri);
        if (tableName.isEmpty()) {
            throw new IllegalArgumentException("Unsupport URI: " + uri);
        }
        int row = mDatabase.update(tableName, values, selection, selectionArgs);
        if (row > 0) {
            mContext.getContentResolver().notifyChange(uri, null);
        }
        return row;
    }
}
//  注冊內(nèi)容提供者
<provider
            android:name="com.rtfsc.CustomProvider"
            android:authorities="com.rtfsc.auth.provider"
            android:permission="com.rtfsc.provider"
            android:process=":provider"/>

Client


//  添加權(quán)限
<uses-permission android:name="com.rtfsc.provider"/>
public class MainActivity extends AppCompatActivity {

    public static final String PROVIDER_CONTENT_PREFIX = "content://com.rtfsc.auth.provider";

    private IOnChangeListener mListener = new IOnChangeListener.Stub() {
        @Override
        public void onChange(String aString) throws RemoteException {
            System.out.println(aString);
        }
    };

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Uri bookUri = Uri.parse(PROVIDER_CONTENT_PREFIX + "/book");
        Uri userUri = Uri.parse(PROVIDER_CONTENT_PREFIX + "/user");

        ContentValues contentValues = new ContentValues();
        contentValues.put("_id", 4);
        contentValues.put("name", "Effctive of java");

        getContentResolver().insert(bookUri, contentValues);

        Cursor cursor = getContentResolver().query(bookUri, new String[]{"_id", "name"}, null, null, null);
        if (cursor == null) {
            return;
        }
        while (cursor.moveToNext()) {
            String id = cursor.getString(cursor.getColumnIndex("_id"));
            String name = cursor.getString(cursor.getColumnIndex("name"));
            System.out.println("id = " + id + " name = " + name);
        }
        cursor.close();
    }
}
//  log
07-29 09:28:34.885 1277-1277/com.rtfsc I/System.out: id = 1 name = Android
07-29 09:28:34.885 1277-1277/com.rtfsc I/System.out: id = 2 name = iOS
07-29 09:28:34.885 1277-1277/com.rtfsc I/System.out: id = 3 name = Pytnon
07-29 09:28:34.885 1277-1277/com.rtfsc I/System.out: id = 4 name = Effctive of java
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末艾凯,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子懂傀,更是在濱河造成了極大的恐慌览芳,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,402評論 6 499
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件鸿竖,死亡現(xiàn)場離奇詭異,居然都是意外死亡铸敏,警方通過查閱死者的電腦和手機(jī)缚忧,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,377評論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來杈笔,“玉大人闪水,你說我怎么就攤上這事∶删撸” “怎么了球榆?”我有些...
    開封第一講書人閱讀 162,483評論 0 353
  • 文/不壞的土叔 我叫張陵朽肥,是天一觀的道長。 經(jīng)常有香客問我持钉,道長衡招,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,165評論 1 292
  • 正文 為了忘掉前任每强,我火速辦了婚禮始腾,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘空执。我一直安慰自己浪箭,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,176評論 6 388
  • 文/花漫 我一把揭開白布辨绊。 她就那樣靜靜地躺著奶栖,像睡著了一般。 火紅的嫁衣襯著肌膚如雪门坷。 梳的紋絲不亂的頭發(fā)上宣鄙,一...
    開封第一講書人閱讀 51,146評論 1 297
  • 那天,我揣著相機(jī)與錄音拜鹤,去河邊找鬼框冀。 笑死,一個(gè)胖子當(dāng)著我的面吹牛敏簿,可吹牛的內(nèi)容都是我干的明也。 我是一名探鬼主播,決...
    沈念sama閱讀 40,032評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼惯裕,長吁一口氣:“原來是場噩夢啊……” “哼温数!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起蜻势,我...
    開封第一講書人閱讀 38,896評論 0 274
  • 序言:老撾萬榮一對情侶失蹤撑刺,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后握玛,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體够傍,經(jīng)...
    沈念sama閱讀 45,311評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,536評論 2 332
  • 正文 我和宋清朗相戀三年挠铲,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了冕屯。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,696評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡拂苹,死狀恐怖安聘,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤浴韭,帶...
    沈念sama閱讀 35,413評論 5 343
  • 正文 年R本政府宣布丘喻,位于F島的核電站,受9級特大地震影響念颈,放射性物質(zhì)發(fā)生泄漏泉粉。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,008評論 3 325
  • 文/蒙蒙 一舍肠、第九天 我趴在偏房一處隱蔽的房頂上張望搀继。 院中可真熱鬧,春花似錦翠语、人聲如沸叽躯。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽点骑。三九已至,卻和暖如春谍夭,著一層夾襖步出監(jiān)牢的瞬間黑滴,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,815評論 1 269
  • 我被黑心中介騙來泰國打工紧索, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留袁辈,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,698評論 2 368
  • 正文 我出身青樓珠漂,卻偏偏與公主長得像晚缩,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子媳危,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,592評論 2 353

推薦閱讀更多精彩內(nèi)容