ContentProvider

前言

  • ContentProvider 是 android 提供的專門用于不同應(yīng)用之間進(jìn)行數(shù)據(jù)共享的方式鼻疮。其內(nèi)部也是通過 Binder 機(jī)制實(shí)現(xiàn)。
  • 系統(tǒng)預(yù)設(shè)了許多 ContentProvider,如通訊錄 日程表信息砾嫉。要進(jìn)行跨進(jìn)程訪問這些信息杉编,只需通過 ContentResolver 的 query update insert 和 delete方法(都運(yùn)行在 Binder 線程池中)即可。
  • ContentProvider 主要以表格的形式來組織數(shù)據(jù)赌厅,并可以包含多個(gè)表穷绵。除表格,ContentProvider 還支持文件數(shù)據(jù),如圖片特愿,視頻等仲墨。
  • ContentProvider(內(nèi)容提供者)的scheme已經(jīng)由Android所規(guī)定為:content://

方法說明

/**
*   第一個(gè)參數(shù)是要查詢的provider的Uri。
*   第二個(gè)參數(shù)是要返回的數(shù)據(jù)列的名字揍障,即想要返回表的哪些列目养。如果返回全部列,就置為null毒嫡“┮希可以理解為對(duì)列的過濾。
*   第三個(gè)參數(shù)可以理解為對(duì)行的過濾兜畸。相當(dāng)于sql的where語(yǔ)句努释,如果返回全部行,置為null咬摇。
*   第四個(gè)參數(shù)伐蒂,selection argment。
*   第五個(gè)參數(shù)菲嘴,對(duì)返回行的排序饿自。舉例說明
**/
public Cursor query(Uri uri, String[] projection, String selection,String[] selectionArgs, String sortOrder) {...}

如:
查詢sim卡中含有某個(gè)字的模糊匹配

public ArrayList<Contact_Object> getPhoneNumber(String name, Context context) {
    ArrayList<Contact_Object> arrayList = new ArrayList<Contact_Object>();
    String ret = null;
    String selection = ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME+" like'" + name +"%'";
    String[] projection = new String[] {ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME, ContactsContract.CommonDataKinds.Phone.NUMBER,ContactsContract.CommonDataKinds.Phone.TYPE,ContactsContract.CommonDataKinds.Phone.PHOTO_URI};
    Cursor cur = context.getContentResolver().query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI, projection, selection, null, null);
    while (cur.moveToNext())
    {
        Contact_Object contactObject = new Contact_Object();
        contactObject.name =  cur.getString(0);
        contactObject.mobileNumber =  cur.getString(1);
        contactObject.type =  cur.getInt(2);
        contactObject.uri =  cur.getString(3);
        arrayList.add(contactObject);

    }

    cur.close();
    return arrayList;
}

示例

  1. 聲明自定義的 ContentProvider
public class BookProvider extends ContentProvider {
    //ContentProvider唯一標(biāo)識(shí)
    public static final String AUTHORITY = "com.ryg.chapter_2.book.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);

    static {
        //public void addURI(String authority, String path, int code)
        sUriMatcher.addURI(AUTHORITY, "book", BOOK_URI_CODE);
        sUriMatcher.addURI(AUTHORITY, "user", USER_URI_CODE);
    }

    private Context mContext;
    private SQLiteDatabase mDb;
    
    @Override
    public boolean onCreate() {     //main thread
        mContext = getContext();
        initProviderData();
        return true;
    }
    //初始化放入數(shù)據(jù)
    private void initProviderData() {
        mDb = new DbOpenHelper(mContext).getWritableDatabase();
        
        mDb.execSQL("delete from " + DbOpenHelper.BOOK_TABLE_NAME);
        mDb.execSQL("delete from " + DbOpenHelper.USER_TALBE_NAME);
        mDb.execSQL("insert into book values(3,'Android');");
        mDb.execSQL("insert into book values(4,'Ios');");
        mDb.execSQL("insert into book values(5,'Html5');");
        mDb.execSQL("insert into user values(1,'jake',1);");
        mDb.execSQL("insert into user values(2,'jasmine',0);");
    }

    //運(yùn)行在Binder線程池中,可能每次運(yùn)行的線程不一樣龄坪。
    @Override
    public Cursor query(Uri uri, String[] projection, String selection,
            String[] selectionArgs, String sortOrder) {
        Log.d(TAG, "query, current thread:" + Thread.currentThread().getName());    
        String table = getTableName(uri);
        if (table == null) {throw new IllegalArgumentException("Unsupported URI: " + uri);}
        return mDb.query(table, projection, selection, selectionArgs, null, null, sortOrder, null);
    }

    @Override
    public String getType(Uri uri) {return null;}

    @Override
    public Uri insert(Uri uri, ContentValues values) {
        String table = getTableName(uri);
        if (table == null) { throw new IllegalArgumentException("Unsupported URI: " + uri); }
        mDb.insert(table, null, values);
        //通知外界當(dāng)前的 ContentProvider 中數(shù)據(jù)發(fā)生了變化
        mContext.getContentResolver().notifyChange(uri, null);
        return uri;
    }

    @Override
    public int delete(Uri uri, String selection, String[] selectionArgs) {
        String table = getTableName(uri);
        if (table == null) { throw new IllegalArgumentException("Unsupported URI: " + uri); }
        int count = mDb.delete(table, selection, selectionArgs);
        //通知外界當(dāng)前的 ContentProvider 中數(shù)據(jù)發(fā)生了變化
        if (count > 0) { getContext().getContentResolver().notifyChange(uri, null); }   
        return count;
    }

    @Override
    public int update(Uri uri, ContentValues values, String selection,
            String[] selectionArgs) {
        String table = getTableName(uri);
        if (table == null) { throw new IllegalArgumentException("Unsupported URI: " + uri); }
        int row = mDb.update(table, values, selection, selectionArgs);
        //通知外界當(dāng)前的 ContentProvider 中數(shù)據(jù)發(fā)生了變化
        if (row > 0) { getContext().getContentResolver().notifyChange(uri, null); }
        return row;
    }

    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_TALBE_NAME;
            break;
        }
        return tableName;
    }
}
  1. 數(shù)據(jù)庫(kù)幫助類 DbOpenHelper
public class DbOpenHelper extends SQLiteOpenHelper {
    private static final String DB_NAME = "book_provider.db";
    public static final String BOOK_TABLE_NAME = "book";
    public static final String USER_TALBE_NAME = "user";

    private static final int DB_VERSION = 3;

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

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

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

    @Override   
    public void onCreate(SQLiteDatabase db) {   //如果創(chuàng)建了昭雌,就不會(huì)回調(diào)這個(gè)方法。
        db.execSQL(CREATE_BOOK_TABLE);
        db.execSQL(CREATE_USER_TABLE);
    }

    @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
        // TODO ignored
    }
}
  1. 使用
public class ProviderActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_provider);
        // Uri uri = Uri.parse("content://com.ryg.chapter_2.book.provider");
        // getContentResolver().query(uri, null, null, null, null);
        // getContentResolver().query(uri, null, null, null, null);
        //查詢book表
        Uri bookUri = Uri.parse("content://com.ryg.chapter_2.book.provider/book");
        ContentValues values = new ContentValues();
        values.put("_id", 6);
        values.put("name", "程序設(shè)計(jì)的藝術(shù)");
        getContentResolver().insert(bookUri, values);
        Cursor bookCursor = getContentResolver().query(bookUri, new String[]{"_id", "name"}, null, null, null);
        while (bookCursor.moveToNext()) {
            Book book = new Book();
            book.bookId = bookCursor.getInt(0);
            book.bookName = bookCursor.getString(1);
        }
        bookCursor.close();
        //查詢User表
        Uri userUri = Uri.parse("content://com.ryg.chapter_2.book.provider/user");
        Cursor userCursor = getContentResolver().query(userUri, new String[]{"_id", "name", "sex"}, null, null, null);
        while (userCursor.moveToNext()) {
            User user = new User();
            user.userId = userCursor.getInt(0);
            user.userName = userCursor.getString(1);
            user.isMale = userCursor.getInt(2) == 1;
        }
        userCursor.close();
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末健田,一起剝皮案震驚了整個(gè)濱河市烛卧,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌妓局,老刑警劉巖总放,帶你破解...
    沈念sama閱讀 210,978評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異好爬,居然都是意外死亡局雄,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,954評(píng)論 2 384
  • 文/潘曉璐 我一進(jìn)店門存炮,熙熙樓的掌柜王于貴愁眉苦臉地迎上來炬搭,“玉大人蜈漓,你說我怎么就攤上這事」” “怎么了融虽?”我有些...
    開封第一講書人閱讀 156,623評(píng)論 0 345
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)灼芭。 經(jīng)常有香客問我有额,道長(zhǎng),這世上最難降的妖魔是什么彼绷? 我笑而不...
    開封第一講書人閱讀 56,324評(píng)論 1 282
  • 正文 為了忘掉前任巍佑,我火速辦了婚禮,結(jié)果婚禮上寄悯,老公的妹妹穿的比我還像新娘句狼。我一直安慰自己,他們只是感情好热某,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,390評(píng)論 5 384
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著胳螟,像睡著了一般昔馋。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上糖耸,一...
    開封第一講書人閱讀 49,741評(píng)論 1 289
  • 那天秘遏,我揣著相機(jī)與錄音,去河邊找鬼嘉竟。 笑死邦危,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的舍扰。 我是一名探鬼主播倦蚪,決...
    沈念sama閱讀 38,892評(píng)論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼边苹!你這毒婦竟也來了陵且?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,655評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤个束,失蹤者是張志新(化名)和其女友劉穎慕购,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體茬底,經(jīng)...
    沈念sama閱讀 44,104評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡沪悲,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,451評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了阱表。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片殿如。...
    茶點(diǎn)故事閱讀 38,569評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡贡珊,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出握截,到底是詐尸還是另有隱情飞崖,我是刑警寧澤,帶...
    沈念sama閱讀 34,254評(píng)論 4 328
  • 正文 年R本政府宣布谨胞,位于F島的核電站固歪,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏胯努。R本人自食惡果不足惜牢裳,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,834評(píng)論 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望叶沛。 院中可真熱鬧蒲讯,春花似錦、人聲如沸灰署。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,725評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)溉箕。三九已至晦墙,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間肴茄,已是汗流浹背晌畅。 一陣腳步聲響...
    開封第一講書人閱讀 31,950評(píng)論 1 264
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留寡痰,地道東北人抗楔。 一個(gè)月前我還...
    沈念sama閱讀 46,260評(píng)論 2 360
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像拦坠,于是被迫代替她去往敵國(guó)和親连躏。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,446評(píng)論 2 348

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

  • 內(nèi)容提供器 思維導(dǎo)圖 最近在練習(xí)使用思維導(dǎo)圖贪婉,獻(xiàn)一下丑反粥。。疲迂。 什么是內(nèi)容提供器 內(nèi)容提供器主要用于在不同應(yīng)用程序間...
    HeilHelloWorld閱讀 785評(píng)論 2 4
  • #Android 基礎(chǔ)知識(shí)點(diǎn)總結(jié) ---------- ##1.adb - android debug bridg...
    Mythqian閱讀 3,264評(píng)論 2 11
  • 簡(jiǎn)評(píng):區(qū)塊鏈如今大火大熱尤蒿,如果你還不知道什么是區(qū)塊鏈郑气,可以通過這 50 行代碼來幫助你的理解。 區(qū)塊鏈 在更通用的...
    極小光閱讀 717評(píng)論 0 5
  • 想起前一陣子去參加一位導(dǎo)師的婚禮腰池,說實(shí)話尾组,并沒有我想象中的那種美好的感覺忙芒。像是,嫁給了生活讳侨。 因?yàn)闀r(shí)間總是不等人...
    SHL先森家的貓閱讀 328評(píng)論 0 0
  • 寫東西的目的是讓自己糾正所有錯(cuò)誤的想法呵萨,錯(cuò)誤的做法,樹立正確的想法與行動(dòng)跨跨!不是消耗自己的時(shí)間潮峦,既然寫了就要堅(jiān)持下去...
    信澤7閱讀 136評(píng)論 0 0