ContentProvider

前言

  • ContentProvider 是 android 提供的专门用于不同应用之间进行数据共享的方式。其内部也是通过 Binder 机制实现。
  • 系统预设了许多 ContentProvider,如通讯录 日程表信息。要进行跨进程访问这些信息,只需通过 ContentResolver 的 query update insert 和 delete方法(都运行在 Binder 线程池中)即可。
  • ContentProvider 主要以表格的形式来组织数据,并可以包含多个表。除表格,ContentProvider 还支持文件数据,如图片,视频等。
  • ContentProvider(内容提供者)的scheme已经由Android所规定为:content://

方法说明

/**
*   第一个参数是要查询的provider的Uri。
*   第二个参数是要返回的数据列的名字,即想要返回表的哪些列。如果返回全部列,就置为null。可以理解为对列的过滤。
*   第三个参数可以理解为对行的过滤。相当于sql的where语句,如果返回全部行,置为null。
*   第四个参数,selection argment。
*   第五个参数,对返回行的排序。举例说明
**/
public Cursor query(Uri uri, String[] projection, String selection,String[] selectionArgs, String sortOrder) {...}

如:
查询sim卡中含有某个字的模糊匹配

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唯一标识
    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;
    }
    //初始化放入数据
    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);");
    }

    //运行在Binder线程池中,可能每次运行的线程不一样。
    @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);
        //通知外界当前的 ContentProvider 中数据发生了变化
        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);
        //通知外界当前的 ContentProvider 中数据发生了变化
        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);
        //通知外界当前的 ContentProvider 中数据发生了变化
        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. 数据库帮助类 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) {   //如果创建了,就不会回调这个方法。
        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", "程序设计的艺术");
        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();
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容

  • 内容提供器 思维导图 最近在练习使用思维导图,献一下丑。。。 什么是内容提供器 内容提供器主要用于在不同应用程序间...
    HeilHelloWorld阅读 783评论 2 4
  • #Android 基础知识点总结 ---------- ##1.adb - android debug bridg...
    Mythqian阅读 3,249评论 2 11
  • 简评:区块链如今大火大热,如果你还不知道什么是区块链,可以通过这 50 行代码来帮助你的理解。 区块链 在更通用的...
    极小光阅读 714评论 0 5
  • 想起前一阵子去参加一位导师的婚礼,说实话,并没有我想象中的那种美好的感觉。像是,嫁给了生活。 因为时间总是不等人...
    SHL先森家的猫阅读 326评论 0 0
  • 写东西的目的是让自己纠正所有错误的想法,错误的做法,树立正确的想法与行动!不是消耗自己的时间,既然写了就要坚持下去...
    信泽7阅读 135评论 0 0