1. 介绍

android中有个ContentProvider(内容提供者)还有个ContentResolver(内容解析者)

简单来说,contentProvider就是将自己app的数据库共享,使得其他app可以访问本app的数据,而contentResolver就是用来访问其他app的数据库的。contentProvider负责组织应用程序的数据,向其他应用程序提供数据;contentResolver则负责获取contentProvider提供的数据,修改/添加/删除/更新数据等。

如何将应用程序的数据暴露出去? Android提供了ContentProvider,一个程序可以通过实现一个Content provider的抽象接口将自己的数据完全暴露出去,而且Content providers是以类似数据库中表的方式将数据暴露。Content providers存储和检索数据,通过它可以让所有的应用程序访问到,这也是应用程序之间唯一共享数据的方法。要想使应用程序的数据公开化,可通过2种方法:创建一个属于你自己的Content provider或者将你的数据添加到一个已经存在的Content provider中,前提是有相同数据类型并且有写入Content provider的权限。

如何通过一套标准及统一的接口获取其他应用程序暴露的数据?Android提供了 ContentResolver,外界的程序可以通过ContentResolver接口访问ContentProvider提供的数据。

当前篇主要说明,如何获取其它应用程序共享的数据,比如获取Android 手机电话薄中的信息。

下面看一则使用contentResolver实例:

获取手机的联系人及头像

public static void getContactsInfo(Context context) {

        //1. 获取内容解析者

        ContentResolver contentResolver = context.getContentResolver();

        // ContactsContract 联系人的API

        Uri uri = ContactsContract.CommonDataKinds.Phone.CONTENT_URI;

        String[] projection = new String[]{

                ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME,

                ContactsContract.CommonDataKinds.Phone.NUMBER,

                ContactsContract.CommonDataKinds.Phone.CONTACT_ID

        };

        //2.查询操作

        //uri               :查询地址

        //projection        :查询的数据字段名称

        //selection         :查询的条件 where id=..

        //selectionArgs     :查询条件的参数

        //sortOrder         :排序s

//        contentResolver.query(uri, projection, selection, selectionArgs, sortOrder);

        Cursor cursor = contentResolver.query(uri, projection, null, null, null);

        while (cursor.moveToNext()) {

            String name = cursor.getString(0);

            String number = cursor.getString(1);

            int id = cursor.getInt(2);

            Bitmap bitmap = getContactPhoto(context, id);

            if (null == bitmap) {

                Log.e("Contact", "id: " + id + " 头像不为空");

            } else {

                Log.e("Contact", "id: " + id + " 头像为空");

            }

        }

    }

    public static Bitmap getContactPhoto(Context context, int id) {

        ContentResolver contentResolver = context.getContentResolver();

//        Uri uri = ContactsContract.Contacts.CONTENT_URI;

        //拼接路径

        //http://www.baidu.com/jdk

        //参数一:表的路径

        //参数二:联系人具体的路径

        Uri uri = Uri.withAppendedPath(ContactsContract.Contacts.CONTENT_URI, id + "");

        //获取联系人头像,以流的方式返回

        InputStream is = ContactsContract.Contacts.openContactPhotoInputStream(contentResolver, uri);

        Bitmap bitmap = BitmapFactory.decodeStream(is);

        if (is != null) {

            try {

                is.close();

                is = null;

            } catch (IOException e) {

                e.printStackTrace();

            }

        }

        return bitmap;

    }

上面就是获取手机联系人的代码了,也是一个内容解析者的使用实例,所要注意的是URI,URI使用来区分不同应用程序的。

关于ContactsContract这个类,由于Android的碎片化极其严重,各个手机品牌有不同的ROOM,这使得Android的软件开发不容易做适配,Google已经想到这一点了,定义了一个标准,所有的厂家都遵循这个标准,而对于我们开发者来说只需要使用这个Api就行了,而这个ContactsContract类就是Google用来适配的,通过这个类我们可以来获取手机中的联系人,而不用担心适配问题。

ContentResolver是安卓中访问媒体的工具类,通过ContentResolver resolver = context.getContentResolver();可以获取这个类的对象,context是上下文对象,一般来说context就是activity。

ContentResolver的使用像数据库的使用,有增删改查四个常用操作。

比如下面这段代码就可以删除手机上一张照片:

ContentResolver resolver = context.getContentResolver();

                    resolver.delete(

                            MediaStore.Images.Media.EXTERNAL_CONTENT_URI,

                            MediaStore.Images.Media.DISPLAY_NAME + "='" + originFileName + "." + photo.extensionName + "'",

                            null

                    );

可以把MediaStore.Images.Media.EXTERNAL_CONTENT_URI理解为数据库中的表。

可以把MediaStore.Images.Media.DISPLAY_NAME + "='" + originFileName + "." + photo.extensionName + "'"理解为数据库的where条件语句。

2、什么是URI? 
在学习如何获取ContentResolver前,有个名词是必须了解的:URI。URI是网络资源的定义,在Android中赋予其更广阔的含义。

将其分为A,B,C,D 4个部分: (四个参数)
A:标准前缀,用来说明一个Content Provider控制这些数据,无法改变的;
B:URI的标识,它定义了是哪个Content Provider提供这些数据。对于第三方应用程序,为了保证URI标识的唯一性,它必须是一个完整的、小写的  类名。这个标识在<provider> 元素的 authorities属性中说明:
<provider name=”.TransportationProvider”  authorities=”com.example.transportationprovider”  . . .  >
C:路径,Content Provider使用这些路径来确定当前需要生什么类型的数据,URI中可能不包括路径,也可能包括多个;
D:如果URI中包含,表示需要获取的记录的ID;如果没有ID,就表示返回全部;
由于URI通常比较长,而且有时候容易出错,切难以理解。所以,在Android当中定义了一些辅助类,并且定义了一些常量来代替这些长字符串,例如:People.CONTENT_URI

3ContentResolver 介绍说明 

看完这些介绍,大家一定就明白了,ContentResolver是通过URI来查询ContentProvider中提供的数据。除了URI以外,还必须知道需要获取的数据段的名称,以及此数据段的数据类型。如果你需要获取一个特定的记录,你就必须知道当前记录的ID,也就是URI中D部分。

前面也提到了Content providers是以类似数据库中表的方式将数据暴露出去,那么ContentResolver也将采用类似数据库的操作来从Content providers中获取数据。现在简要介绍ContentResolver的主要接口,如下:

返回值

函数声明

final Uri

insert(Uri url, ContentValues values)Inserts a row  into a table at the given URL.

final int

delete(Uri url, String where, String[] selectionArgs)Deletes row(s)  specified by a content URI.

final Cursor

query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) Query the given  URI, returning a Cursor over the result set.

final int

update(Uri uri, ContentValues values, String where, String[] selectionArgs)Update row(s) in  a content URI.

知道ContentResolver是通过ContentProvider来获取其他与应用程序共享的数据,那么ContentResolver与ContentProvider的接口应该差不多的。

4、ContentProvider 是如何向外界提供数据的?
Android提供了ContentProvider,一个程序可以通过实现一个ContentProvider的抽象接口将自己的数据完全暴露出去,而且ContentProviders是以类似数据库中表的方式将数据暴露,也就是说ContentProvider就像一个“数据库”。那么外界获取其提供的数据,也就应该与从数据库中获取数据的操作基本一样,只不过是采用URI来表示外界需要访问的“数据库”。

简要分析下ContentProvider向外界提供数据操作的接口:
query(Uri, String[], String, String[], String)
insert(Uri, ContentValues)
update(Uri, ContentValues, String, String[])
delete(Uri, String, String[])

这些操作与数据库的操作基本完全一样

在URI的D部分可能包含一个_ID ,这个应该出现在SQL语句中的,可以以种特殊的方式出现,这就要求我们在提供数据的时候,需要来额外关注这个特殊的信息。Android  SDK推荐的方法是:在提供数据表字段中包含一个ID,在创建表时INTEGER PRIMARY KEY AUTOINCREMENT标识此ID字段。

5、ContentProvider 是如何组织数据的?
组织数据主要包括:存储数据,读取数据,以数据库的方式暴露数据。数据的存储需要根据设计的需求,选择合适的存储结构,首选数据库,当然也可以选择本地其他文件,甚至可以是网络上的数据。数据的读取,以数据库的方式暴露数据这就要求,无论数据是如何存储的,数据最后必须以数据的方式访问。

可能还有2个问题,是需要关注的。

  1. ContentProvider是什么时候创建的,是谁创建的?访问某个应用程序共享的数据,是否需要启动这个应用程序?这个问题在 Android SDK中没有明确说明,但是从数据共享的角度出发,ContentProvider应该是Android在系统启动时就创建了,否则就谈不上数据共享了。这就要求在AndroidManifest.XML中使用<provider>元素明确定义。
  2. 可能会有多个程序同时通过ContentResolver访问一个ContentProvider,会不会导致像数据库那样的“脏数据”?这个问题一方面需要数据库访问的同步,尤其是数据写入的同步,在AndroidManifest.XML中定义ContentProvider的时候,需要考虑是<provider>元素multiprocess属性的值;另外一方面Android在ContentResolver中提供了 notifyChange()接口,在数据改变时会通知其他ContentObserver,这个地方应该使用了观察者模式,在 ContentResolver中应该有一些类似register,unregister的接口。

至此,已经对ContentProvider提供了比较全面的分析,至于如何创建ContentProvider,可通过2种方法:创建一个属于你自己的ContentProvider或者将你的数据添加到一个已经存在的ContentProvider中,当然前提是有相同数据类型并且有写入 Content provider的权限。

Logo

为开发者提供学习成长、分享交流、生态实践、资源工具等服务,帮助开发者快速成长。

更多推荐