http://blog.csdn.net/zuolongsnail/article/details/6566317

ContentProvider简介

ContentProvider(内容提供者)主要用于对外共享数据,也就是通过ContentProvider把应用中的数据共享给其他应用访问,其他应用可以通过ContentProvider对指定应用中的数据进行操作。ContentProvider分为系统的和自定义的,系统的也就是例如联系人,照片等数据。使用ContentProvider对外共享数据的好处是统一了数据的访问方式


1.Uri

ContentResolver中的增删改查方法都是不接收表名参数的,而是使用一个Uri参数代替,这个参数被称为内URI。内容URI给内容提供器中的数据建立了唯一标识符,它主要由两部分组成,权限(authority)和路径(path)。权限是用于对不同的应用程序做区分的。

内容 URI最标准的格式写法如下:content://com.example.app.provider/table1;

上面的 URI可以非常清楚地表达出我们想要访问com.example.app.providertable1这张表里的数据

在得到了内容 URI字符串之后,我们还需要将它解析成 Uri对象才可以作为参数传入。解析的方法也相当简单,代码如下所示:Uri uri = Uri.parse("content://com.example.app.provider/table1")



2.ContentResolver

对于每一个应用程序来说,如果想要访问内容提供器中共享的数据,就一定要借助ContentResolve类,可以通过 Context中的 getContentResolver()方法获取到该类的实例。ContentResolver中提供了一系列的方法用于对数据进行增删改查操作,其中 insert()方法用于添加数据,update()方法用于更新数据,delete()方法用于删除数据,query()方法用于查询数据。

public Uri insert(Uri uri, ContentValues values)
public int delete(Uri uri, String selection, String[] selectionArgs)
public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs)
public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder)

现在我们就可以使用这个 Uri对象来查询 table1表中的数据了,代码如下所示:
Cursor cursor = getContentResolver().query(uri,projection,selection,selectionArgs,sortOrder);

如何向 table1表中添加一条数据,代码如下所示:

ContentValues values = new ContentValues();
values.put("column1", "text");
values.put("column2", 1);
getContentResolver().insert(uri, values)

如何 更新 这 条 新 添加 的 数 据 ,把 column1 的 值 清空 , 可 以 借 助 ContentResolver 的 update() 方法实现,代码如下:

ContentValues values = new ContentValues();
values.put("column1", "");
getContentResolver().update(uri, values, "column1 = ? and column2 =?", new String[] {"text", "1"});

注意上述代码使用了 selection 和 selectionArgs 参数来对想要更新的数据进行约束,以防止所有的行都会受影响。

最后,可以调用 ContentResolver的 delete()方法将这条数据删除掉,代码如下所示:
getContentResolver().delete(uri, "column2 = ?", new String[] {"1" });

3.ContentProvider

创建内容提供器的步骤(参考至:《第一行代码》)
如果想要把自己写的数据实现跨程序共享,可以通过新建一个 ContentProvider 子类的方式来创建一个自己的内容提供器。ContentProvider类中有六个抽象方法,我们在使用子类继承它的时候,需要将这六个方法全部重写。

新建 MyProvider继承自 ContentProvider,代码如下所示:

public class MyProvider extends ContentProvider {@Override
    public boolean onCreate() {return false;
    }@Override
    public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {return null;
    }@Override
    public String getType(Uri uri) {return null;
    }@Override
    public Uri insert(Uri uri, ContentValues values) {return null;
    }@Override
    public int delete(Uri uri, String selection, String[] selectionArgs) {return 0;
    }@Override
    public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {return 0;
    }
}

1. onCreate()
初始化内容提供器的时候调用。通常会在这里完成对数据库的创建和升级等操作, 返回 true 表示内容提供器初始化成功,返回 false 则表示失败。注意,只有当存在 ContentResolver 尝试访问我们程序中的数据时,内容提供器才会被初始化。
2. query()
从内容提供器中查询数据。使用 uri 参数来确定查询哪张表, projection 参数用于确 定查询哪些列, selection 和 selectionArgs 参数用于约束查询哪些行, sortOrder 参数用于 对结果进行排序,查询的结果存放在 Cursor 对象中返回。
3. insert()
向内容提供器中添加一条数据。使用 uri 参数来确定要添加到的表,待添加的数据 保存在 values 参数中。添加完成后,返回一个用于表示这条新记录的 URI 。
4. update()
更新内容提供器中已有的数据。使用 uri 参数来确定更新哪一张表的数据,新数 据保存在 values 参数中, selection 和 selectionArgs 参数用于约束更新哪些行,受影响的 行数将作为返回值返回。
5. delete()
从内容提供器中删除数据。使用 uri参数来确定删除哪一张表中的数据,selection和 selectionArgs 参数用于约束删除哪些行,被删除的行数将作为返回值返回

6.getType()
根据传入的内容 URI来返回相应的 MIME类型。可以看到,几乎每一个方法都会带有 Uri这个参数,这个参数也正是调用 ContentResolver的增删改查方法时传递过来的。而现在,我们需要对传入的 Uri 参数进行解析,从中分析出调用方期望访问的表和数据。

在Manifest.xml文件中注册:

<application
   。。。。
    <provider
        android:name=".MyProvider"
        android:authorities="com.example.app.provider"
        android:enabled="true"
        android:exported="true"/>
</application>

android:authorities属性中指定了该内容提供器的权限。上面的url知识中,uri由权限(authority)和路径(path)组成,这里说权限(authority)就是的android:authorities属性的值。比如:content://com.example.app.provider/table1中的“com.example.app.provider”;

android:exported 是Android中的四大组件 Activity,Service,Provider,Receiver 四大组件中都会有的一个属性。总体来说它的主要作用是:是否支持其它应用调用当前组件。 默认值:如果包含有intent-filter 默认值为true; 没有intent-filter默认值为false。如果设置为true,则能够被调用或交互。设置为false时,只有同一个应用程序的组件或拥有同样user ID的Application的组件调用或者启动。

回顾一下,一个标准的内容 URI写法是这样的:content://com.example.app.provider/table1这就表示调用者期望访问的是authorities=“com.example.app.provider”的ContentProvider操作的数据库里面的table1表中的数据。除此之外,我们还可以在这个内容 URI的后面加上一个 id,如下所示:content://com.example.app.provider/table1/1,这就表示调用方期望访问的是 table1表中 id为 1的数据。
      内容 URI的格式主要就只有以上两种,以路径结尾就表示期望访问该表中所有的数据,以 id结尾就表示期望访问该表中拥有相应 id的数据。我们可以使用通配符的方式来分别匹配这两种格式的内容 URI,规则如下。
1. *:表示匹配任意长度的任意字符
2. #:表示匹配任意长度的数字
所以,一个能够匹配任意表的内容 URI格式就可以写成:content://com.example.app.provider/*
而一个能够匹配 table1表中任意一行数据的内容 URI格式就可以写成:content://com.example.app.provider/table1/#

接着,我们再借助UriMatcher这个类就可以轻松地实现匹配内容 URI的功能。

因为Uri代表了要操作的数据,所以我们很经常需要解析Uri,并从Uri中获取数据。Android系统提供了两个用于操作Uri的工具类,分别为UriMatcher和ContentUris。掌握它们的使用,会便于我们的开发工作

UriMatcher类用于匹配Uri,它的用法如下:

UriMatcher uriMatcher= new UriMatcher(UriMatcher.NO_MATCH);
//匹配码是调用addURI()方法传入的第三个参数
uriMatcher.addURI(AUTHORITY, TABLE_BOOK, BOOK_DIR);
uriMatcher.addURI(AUTHORITY, TABLE_BOOK + "/#", BOOK_ITEM);//#号为通配符,有参数,匹配book——ITEM

1)UriMatcher  uriMatcher = new UriMatcher(UriMatcher.NO_MATCH)// /常量UriMatcher.NO_MATCH表示不匹配任何路径的返回码 2)uriMather.addURI("com.example.app.provider","book"_BOOK,BOOK_DIR)

int BOOK_DIR = 0; int BOOK_ITEM = 1; //表示匹配码,uriMatcher.match(uri)该方法就会返回匹配码,通过匹配码,就可以方便的分清楚传入uri的人想做什么操作了。

比如传入该uri:Uri uri = Uri.parse("content://"+"com.example.app.provider" + "/" + "book");后,经过uriMatcher.match(uri)匹配,返回码就是BOOK_DIR ,

如果传入的uri是:Uri uri = Uri.parse("content://"+"com.example.app.provider" + "/" + "book"+“/”+"2"),那么uriMatcher.match(uri)匹配后返回的就是BOOK_ITEM,因为“/#”代表通配符,表示有参数就会匹配

uriMatcher.addURI(AUTHORITY, TABLE_BOOK + "/#", BOOK_ITEM)

这一条的匹配码(第三个参数)。

通过UriMatcher我们就可以判断出调用者期望访问的是哪张表中的数据,我们还可以得出他是调用这表中的全部数据呢,还是指定要某一条或者满足某个条件的数据,至于(“/”+"2")这里面的2表示的是id还是age还是其他字段,是由自己和provider提供者定义,两则要统一就可以了,比如说,自定义的ContentProvider在book表中,传入的数字表示id,那么传uri的人,就要传入你想要操作的book的id,而不是其他的,不然也找不到你要的数据。

修改 MyProvider中的代码,如下所示:

public class MyProvider extends ContentProvider {public static final int TABLE1_DIR = 0;
    public static final int TABLE1_ITEM = 1;
    public static final int TABLE2_DIR = 2;
    public static final int TABLE2_ITEM = 3;
    private static UriMatcher uriMatcher;

    static {uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
        uriMatcher.addURI("com.example.app.provider", "table1", TABLE1_DIR);
        uriMatcher.addURI("com.example.app.provider ", "table1/#", TABLE1_ITEM);
        uriMatcher.addURI("com.example.app.provider ", "table2", TABLE2_ITEM);
        uriMatcher.addURI("com.example.app.provider ", "table2/#", TABLE2_ITEM);
    }@Override
    public Cursor query(Uri uri, String[] projection, String selection,
                        String[] selectionArgs, String sortOrder) {switch (uriMatcher.match(uri)) {case TABLE1_DIR:// 查询table1表中的所有数据
                break;
            case TABLE1_ITEM:// 查询table1表中的单条数据
                break;
            case TABLE2_DIR:// 查询table2表中的所有数据
                break;
            case TABLE2_ITEM:// 查询table2表中的单条数据
                break;
            default:break;
        }return  。。。;
    }
    。。。。。
}

可以看到,MyProvider中新增了四个整型常量,其中 TABLE1_DIR 表示访问 table1表中的所有据,TABLE1_ITEM表示访问 table1表中的单条数据,TABLE2_DIR表示访问table2表中的所有数据,TABLE2_ITEM表示访问 table2表中的单条数据。接着在静态代码块里我们创建了 UriMatcher 的实例,并调用 addURI()方法,将期望匹配的内容 URI 格式传递进去,注意这里传入的路径参数是可以使用通配符的。然后当 query()方法被调用的时候,就会通过 UriMatcher 的 match()方法对传入的 Uri 对象进行匹配,如果发现 UriMatcher中某个内容 URI格式成功匹配了该 Uri对象,则会返回相应的自定义代码,然后我们就可以判断出调用方期望访问的到底是什么数据了。
上述代码只是以 query()方法为例做了个示范,其实 insert()、update()、delete()这几个方法的实现也是差不多的,它们都会携带 Uri 这个参数,然后同样利用 UriMatcher的 match()方法判断出调用方期望访问的是哪张表,再对该表中的数据进行相应的操作就可以了。

除此之外,还有一个方法你会比较陌生,即 getType()方法。它是所有的内容提供器都必须提供的一个方法,用于获取 Uri 对象所对应的 MIME类型。一个内容 URI所对应的 MIME字符串主要由三部分组分,Android对这三个部分做了如下格式规定。
1. 必须以 vnd 开头。
2. 如果内容 URI 以路径结尾,则后接 android.cursor.dir/,如果内容 URI以 id结尾,则后android.cursor.item/。
3. 最后接上 vnd.<authority>.<path>。
所以,对于 content://com.example.app.provider/table1这个内容 URI,它所对应的 MIME类型就可以写成:
vnd.android.cursor.dir/vnd.com.example.app.provider.table1
对于 content://com.example.app.provider/table1/1这个内容 URI,它所对应的 MIME 类型就可以写成:
vnd.android.cursor.item/vnd. com.example.app.provider.table1

到这里,一个完整的内容提供器就创建完成了,现在任何一个应用程序都可以使用ContentResolver来访问我们程序中的数据

完整代码:

public class MyTestProvider extends ContentProvider {private static final String AUTHORITY = "com.hujing.myprovider";
    public static final String CONTENT_AUTHORITY = "content://" + AUTHORITY;

    //匹配码   DIR:全部,ITEM:某一条
    public static final int BOOK_DIR = 0;
    public static final int BOOK_ITEM = 1;
    public static final String TABLE_BOOK = "book";

    DataOpenHelper dbHelper;
    private static UriMatcher uriMatcher;

    static {uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
        //匹配码是调用addURI()方法传入的第三个参数
        uriMatcher.addURI(AUTHORITY, TABLE_BOOK, BOOK_DIR);
        uriMatcher.addURI(AUTHORITY, TABLE_BOOK + "/#", BOOK_ITEM);//#号为通配符,有参数,匹配book——ITEM
    }@Override
    public boolean onCreate() {//数据库的创建和升级等操作
        dbHelper = new DataOpenHelper(getContext());
        return true;
    }@Override
    public String getType(Uri uri) {return null;
    }//查询
    @Override
    public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {SQLiteDatabase db = dbHelper.getReadableDatabase();
        Cursor cursor = null;
        switch (uriMatcher.match(uri)) {case BOOK_DIR:cursor = db.query(TABLE_BOOK, projection, selection, selectionArgs, sortOrder, null, null);
                break;
            case BOOK_ITEM:String id = uri.getPathSegments().get(1);
                cursor = db.query(TABLE_BOOK, projection, "id = ?", new String[]{id}, sortOrder, null, null);
                break;
        }return cursor;
    }//单条插入
    @Override
    public Uri insert(Uri uri, ContentValues values) {SQLiteDatabase db = dbHelper.getWritableDatabase();
        Uri insertUri = null;
        switch (uriMatcher.match(uri)) {case BOOK_DIR:case BOOK_ITEM:long bookId = db.insert(TABLE_BOOK, null, values);
                insertUri = Uri.parse("content://" + AUTHORITY + "/" + TABLE_BOOK + "/" + bookId);
                break;
        }return insertUri;
    }//批量插入
    @Override
    public int bulkInsert(Uri uri, ContentValues[] values) {int numValues = 0;
        SQLiteDatabase db = dbHelper.getWritableDatabase();
        db.beginTransaction();
        try {numValues = values.length;
            switch (uriMatcher.match(uri)) {case BOOK_DIR:for (ContentValues contentValues : values) {db.insert(TABLE_BOOK, null, contentValues);
                    }break;
            }db.setTransactionSuccessful();
        } finally {db.endTransaction();
        }return numValues;
    }//删除
    @Override
    public int delete(Uri uri, String selection, String[] selectionArgs) {SQLiteDatabase db = dbHelper.getWritableDatabase();
        int deletedRows = 0;
        switch (uriMatcher.match(uri)) {case BOOK_DIR://删除全部
                deletedRows = db.delete(TABLE_BOOK, selection, selectionArgs);
                break;
            case BOOK_ITEM://删除指定id
                String bookId = uri.getPathSegments().get(1);
// 比如:content://"+AUTHORIY+"/users /1",getPathSegments()得到的是users 和1,get(1)会得到1
                deletedRows = db.delete(TABLE_BOOK, "id = ?", new String[]{bookId});
                break;
        }return deletedRows;
    }//更新某一条
    @Override
    public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {SQLiteDatabase db = dbHelper.getReadableDatabase();
        int i = 0;
        switch (uriMatcher.match(uri)) {case BOOK_DIR:i = db.update(TABLE_BOOK, values, selection, selectionArgs);
                break;
            case BOOK_ITEM:String bookId = uri.getPathSegments().get(1);
                i = db.update(TABLE_BOOK, values, "id = ?", new String[]{ bookId });
                break;
        }return i;
    }
}

比如:获取联系人

import java.util.ArrayList;
import java.util.List;
import com.hujing.android22.test.R;
import com.hujing.android22.test.adapter.DBAdapter;
import com.hujing.android22.test.object.User;
import android.app.Activity;
import android.content.ContentResolver;
import android.database.Cursor;
import android.net.Uri;
import android.os.Bundle;
import android.provider.ContactsContract.CommonDataKinds.Phone;
import android.provider.ContactsContract.Contacts;
import android.util.Log;
import android.widget.ListView;public class ContactsActivity extends Activity {ListView lv;List<User> list;DBAdapter adapter;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.contacts);lv = (ListView) findViewById(R.id.listView1);list = new ArrayList<User>();adapter = new DBAdapter(readContacts(list), this);lv.setAdapter(adapter);//readContacts();}/*** 读取联系人* @param list ,* @return */private List<User> readContacts(List<User> list) {<strong>ContentResolver</strong> res = getContentResolver();Uri uri_tel = Phone.CONTENT_URI;Uri uri = Contacts.CONTENT_URI;Cursor cursor = res.query(uri, null, null, null, null);while (cursor.moveToNext()) {String contacts_name = cursor.getString(cursor.getColumnIndex(Contacts.DISPLAY_NAME));int contacts_id = cursor.getInt(cursor.getColumnIndex(Contacts._ID));Cursor cursor_tel = res.query(uri_tel, null, Phone.CONTACT_ID + "="+ contacts_id, null, null);StringBuffer buffer = new StringBuffer();String number = null;while (cursor_tel.moveToNext()) {number = cursor_tel.getString(cursor_tel.getColumnIndex(Phone.NUMBER));buffer.append(number);buffer.append("/");}cursor_tel.close();User user=new User();user.setId(1);user.setName(contacts_name);user.setTel(buffer.toString());list.add(user);user=null;Log.i("result", contacts_name + buffer);}cursor.close();return list;}
}

android四大组件---ContentProvider相关推荐

  1. Android四大组件ContentProvider

    2019独角兽企业重金招聘Python工程师标准>>> ContentProvider主要用于在不同应用程序之间实现数据共享功能.允许一个程序访问另一个程序,并保证被访数据的安全性. ...

  2. Android基础知识巩固系列 Android之四大组件——ContentProvider(内容提供者)

    因为最近要面试,于是打算整理整理一下Android的基础知识,由于之前本人已经学习过大概的Android基础知识,这里主要讲这四大组件.五大存储.六大布局.网络请求等这些内容,其他一些等有时间再整理, ...

  3. Android四大组件之ContentProvider 全面解析,ContentResolver源码解析如何调用其它APP的ContentProvider

    今天来总结下Android中的ContentProvider(以下简称CP),具体代码请见https://github.com/Mangosir/ContentProviderReview/tree/ ...

  4. Android——四大组件、六大布局、五大存储

    一.android四大组件 (一)android四大组件详解 Android四大组件分别为activity.service.content provider.broadcast receiver. 1 ...

  5. Android 四大组件 与 MVC 架构模式

    作为一个刚从JAVA转过来的Android程序员总会思考android MVC是什么样的? 首先,我们必须得说Android追寻着MVC架构,那就得先说一下MVC是个啥东西! 总体而来说MVC不能说是 ...

  6. android四大组件小整

    原文来自http://www.jianshu.com/p/478a34af17df 所谓的android四大组件一次是Activity.Service.BroadcastReceiver和Conten ...

  7. Android添加手机黑名单,手机来电拦截实现详解与Demo,一个不错的练手项目,涵盖Android四大组件。

    简介 这是一个小应用的详解,这个应用可以添加手机黑名单,拦截手机黑名单的来电.通过这个小demo,我们可以对Android四大组件的应用场景有个具体的了解,可以说是一个不错的练手项目. 下面给出下载地 ...

  8. Android四大组件和启动模式(面试总结)

    一.Android四大组件详解 Android四大组件分别为Activity.Service.Content Provider.Broadcast Receiver. 1.Activity (1)一个 ...

  9. Android四大组件之Content Provider

    Android四大组件之Content Provider 作者:白璐 日期:2020/2/24 文章目录 Android四大组件之Content Provider 概述 内容提供器(Content P ...

最新文章

  1. 七个算法小仙女,写出一本1200页的深度学习技术手册!(限时公开下载)
  2. PHP 接收 UDP包_PHP早已不是十年前的鸟样!!!
  3. 线程池 And 线程池的使用(基于.net平台)
  4. redhat nginx php mysql_redhat7+nginx+mysql+php
  5. 在nodejs中的集成虹软人脸识别
  6. 找字符串中第一个只出现一次的字符
  7. thingsboard官网单机并发量
  8. CG CTF WEB Download~!
  9. VTK:八叉树可视化用法实战
  10. mongodb数据可视化_使用MongoDB实时可视化开放数据
  11. C#LeetCode刷题之#39-组合总和(Combination Sum)
  12. canvas 绘制直线 并选中_在画布中使用路径-Canvas的基本操作
  13. 【机器学习算法应用和学习_1】1.1 机器学习框架
  14. DataGridView列自适应宽度
  15. Maven的安装与配置(详细版)
  16. 碧海威 L7多款产品 后台命令执行漏洞
  17. 小波变换matlab加密,混沌和小波变换的图像加密压缩算法
  18. Python描述 LeetCode 781. 森林中的兔子
  19. 7-26 输出大写英文字母
  20. 向日葵 11.0.0.34335 中文版 (老牌国产远程控制软件)

热门文章

  1. 关于JSONObject调用fromObject方法的出错问题
  2. 算法竞赛入门经典(第二版)自学笔记
  3. 【教程】详解VS2010安装流程
  4. 2016-11-1 python day001
  5. Cesium实现建筑物单体化(分栋分层)
  6. 程序员遇到了技能瓶颈怎么办,我来告诉你突破瓶颈的建议
  7. 墨菲定律:一个参数Drop_caches导致集群数据库实例崩溃
  8. ftp服务器vsftpd配置文件,最全面的vsftpd配置文件详解
  9. MATLAB程序:8PSK 调制解调
  10. 4 模态对话框和非模态对话框