内容提供者(ContentProvider)

Android的四大组件之一,也是最后一个学习的四大组件,ContentProvider的作用是将Android程序的私有数据暴露给其他应用使用,可以自己选择暴露那些使用ContentProvider,一般来说暴露的都是数据库数据

  • 首先要有一个类继承ContentProvider类,重写其中的方法,通关观察重写的方法可以发现,就是制定可以暴露的数据,通过增删改查方法的重写被别的程序调用

    public class Provider extends ContentProvider {
        
        private Sqli sqli;
        private String table="aa";
        
        @Override
        public boolean onCreate() {
            // TODO Auto-generated method stub
            sqli = new Sqli(getContext());
            return false;
        }
    
        @Override
        public Cursor query(Uri uri, String[] projection, String selection,
                String[] selectionArgs, String sortOrder) {
            // TODO Auto-generated method stub
            SQLiteDatabase database = sqli.getWritableDatabase();
            Cursor cursor = database.query(table, null, selection, selectionArgs, null, null, null);
            return cursor;
        }
    
        @Override
        public String getType(Uri uri) {
            // TODO Auto-generated method stub
        
    
            return null;
        }
    
        @Override
        public Uri insert(Uri uri, ContentValues values) {
            // TODO Auto-generated method stub
            SQLiteDatabase database = sqli.getWritableDatabase();
            database.insert(table, null, values);
            
            return null;
        }
    
        @Override
        public int delete(Uri uri, String selection, String[] selectionArgs) {
            // TODO Auto-generated method stub
            SQLiteDatabase database = sqli.getWritableDatabase();
            int delete = database.delete(table, selection, selectionArgs);
            return delete;
        }
    
        @Override
        public int update(Uri uri, ContentValues values, String selection,
                String[] selectionArgs) {
            // TODO Auto-generated method stub
            SQLiteDatabase database = sqli.getWritableDatabase();
            int update = database.update(table, values, selection, selectionArgs);
            return update;
        }
    
  • 使用ContentProvider还必需在清单文件中进行注册

    <?xml version="1.0" encoding="utf-8"?>
    <manifest xmlns:android="http://schemas.android.com/apk/res/android"
        package="com.example.bch_5_22"
        android:versionCode="1"
        android:versionName="1.0" >
    
        <uses-sdk
            android:minSdkVersion="19"
            android:targetSdkVersion="19" />
    
        <application
            android:allowBackup="true"
            android:icon="@drawable/ic_launcher"
            android:label="@string/app_name"
            android:theme="@style/AppTheme" >
            <activity
                android:name=".MainActivity"
                android:label="@string/app_name" >
                <intent-filter>
                    <action android:name="android.intent.action.MAIN" />
    
                    <category android:name="android.intent.category.LAUNCHER" />
                </intent-filter>
            </activity>
            <--! 注册ContentProvider -->
            <provider
                android:name="com.example.pro.Provider"
                android:authorities="com.example.pro.Provider.aa"
                android:exported="true" >
            </provider>
        </application>
    
    </manifest>
    

    android:name="com.example.pro.Provider"

    确定使用的类的路径

    android:authorities="com.example.pro.Provider.aa"

    暴露的uri可以被其他程序调用的接口或者说网址

    android:exported="true"

    是否可以被其他程序调用

内容的解析者ContentResolver

有内容的提供者,就有使用它的东西,这就是内容的解析者ContentResolver,使用ContentProvider,需要两步。

resolver = getContentResolver();
uri = Uri.parse("content://com.example.pro.Provider.aa");
  1. 获取ContentResolver
  2. 获取uri

uri就是内容的提供者暴露的接口,

使用ContentResolver

通过ContentResolver和uri调用ContentProvider的方法获取数据。

package com.example.bch_ss;

import android.app.Activity;
import android.content.ContentResolver;
import android.content.ContentValues;
import android.database.Cursor;
import android.net.Uri;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Toast;

public class MainActivity extends Activity {

    private ContentResolver resolver;
    private Uri uri;

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

        // 获取内容解析者
        resolver = getContentResolver();
        // 指定Uri通过URI访问指定的内容提供者
        uri = Uri.parse("content://com.example.privoide.mycompany");

    }

    public void add(View view) {
        ContentValues values = new ContentValues();
        values.put("name", "小舅子");
        values.put("emp", "经理");
        resolver.insert(uri, values);
    }

    public void find(View view) {
        Cursor cursor = resolver.query(uri, null, null, null, null);
        while (cursor.moveToNext()) {

            Log.i("aaa",
                    "姓名:" + cursor.getString(cursor.getColumnIndex("name"))
                            + "职位:"
                            + cursor.getString(cursor.getColumnIndex("emp")));
        }
    }

    public void updata(View view) {
        // update company set emp = "" where name = ?
        ContentValues values = new ContentValues();
        values.put("emp", "老司机");
        int update = resolver.update(uri, values, "name=?",
                new String[] { "张三" });
        if (update > 0) {
            Toast.makeText(MainActivity.this, "修改成功", 0).show();
        }
    }

    public void del(View view) {
        // delete from company where name = ?
        int delete = resolver.delete(uri, "name=?", new String[] { "小舅子" });
        if (delete > 0) {
            Toast.makeText(MainActivity.this, "删除成功", 0).show();
        }
    }
}

系统提供的ContentProvider

实际上,Android系统为开发者提供了大量的ContentProvider,例如短信信息、联系人信息、系统的多媒体信息等,开发者自己开发的APP也可以通过ContentResolver来间接调用系统提供的ContentProvider所实现的insert()delete()update()query()方法,这样开发者就可以获取到Android内部数据了。

获取联系人

Android系统提供了Contacts应用程序(com.android.providers.contacts)来管理联系人,也为联系人管理提供了相应的ContentProvider,其他应用程序同样可以通过ContentResolver访问管理联系人的ContentProvider,间接操作联系人数据库contacts2.db,即可实现管理联系人。

下面列出了管理联系人主要涉及的联系人数据库中的三张表,以及表中需要关注的字段:

  • raw_contacts表
    • contact_id:联系人ID
  • data表:保存联系人的详细信息,一条信息占一行,而不是一个联系人占一行
    • data1:联系人信息的具体内容
    • raw_contact_id:联系人ID,标识该条信息属于哪个联系人
    • mimetype_id:标识该条信息属于什么MIME类型
  • mimetypes表:保存各个mimetype_id对应的MIME类型(姓名、号码、邮箱...)

下面的代码简单的实现了获取全部联系人到JavaBean并输出

ContentResolver resolver = getContentResolver();
// 先查询raw_contacts表获取所有联系人ID(/raw_contacts表示查询raw_contacts表)
Cursor cursorContactId = 

resolver.query(Uri.parse("content://com.android.contacts/raw_contacts"),
        new String[]{"contact_id"},
        null, null, null);
while (cursorContactId.moveToNext()) {
    String contactId = cursorContactId.getString(0);
    // 使用联系人ID作为where条件去查询属于该联系人的信息(/data实际上是多张表的复合查询)
    Cursor cursorContactInfo = resolver.query(Uri.parse("content://com.android.contacts/data"),
            new String[]{"data1", "mimetype"},
            "raw_contact_id = ?",
            new String[]{contactId}, 
            null);
    ContactBean contact = new ContactBean();    // 联系人JavaBean
    while (cursorContactInfo.moveToNext()) {
        String data1 = cursorContactInfo.getString(0);
        String mimeType = cursorContactInfo.getString(1);
        switch (mimeType) {   // 根据不同的MIME类型把联系人信息保存到JavaBean
            case "vnd.android.cursor.item/name":
                contact.setName(data1);
                break;
            case "vnd.android.cursor.item/phone_v2":
                contact.setPhone(data1);
                break;
            case "vnd.android.cursor.item/email_v2":
                contact.setEmail(data1);
                break;
            default:
                break;
        }
        System.out.println(contact);
    }
}

不要忘记在清单文件中注册读联系人权限

<uses-permission android:name="android.permission.READ_CONTACTS"></uses-permission>

添加联系人

添加联系人,只需要向上面的三张表中插入新联系人的数据。
向raw_contacts表中插入新联系人的contact_id之前,需要先遍历这个表,以确定新联系人的contact_id是多少,注意这里不应该用已有的contact_id(如果用户删除了某个联系人,这个字段会置空)来确定新联系人的contact_id,只能依据主键_id来确定。新联系人的contact_id插入完成之后,我们就可以插入具体的联系人信息。
该过程的完整的代码如下:

 ContentResolver resolver = getContentResolver();
 // 先查询目前最新的联系人的主键_id,新联系人的ID在此基础上加1即可得到
 Cursor cursorId = resolver.query(Uri.parse("content://com.android.contacts/raw_contacts"),
         new String[]{"_id"},
         null, null, null);
 int _id = 0;
 if (cursorId.moveToLast()) {
     _id = cursorId.getInt(0);
 }
 int contactId = _id + 1;
 // 向raw_contacts表中插入新联系人ID
 ContentValues contentValues = new ContentValues();
 contentValues.put("contact_id", contactId);
 resolver.insert(Uri.parse("content://com.android.contacts/raw_contacts"), contentValues);
 
 // 插入具体的联系人信息
 // 插入名字
 contentValues.clear();
 contentValues.put("raw_contact_id", contactId);
 contentValues.put("data1", et_name.getText().toString());
 contentValues.put("mimetype", "vnd.android.cursor.item/name");
 resolver.insert(Uri.parse("content://com.android.contacts/data"), contentValues);
 
 // 插入号码
 contentValues.clear();
 contentValues.put("raw_contact_id", contactId);
 contentValues.put("data1", et_phone.getText().toString());
 contentValues.put("mimetype", "vnd.android.cursor.item/phone_v2");
 resolver.insert(Uri.parse("content://com.android.contacts/data"), contentValues);

这里需要在清单文件中注册读联系人权限和写联系人权限

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

推荐阅读更多精彩内容