ContentResolver与ContentProvider

    技术2022-05-19  39

        写在转帖前:最近遇到个问题,使用ContentResolver插入数据然后取出该条数据,当时是正常的,但当把进程杀掉后,下次启动程序时,程序竟然重新调用ContentProvider新建数据库,导致数据丢失。以下仅是明确两者使用方法并不是问题答案。

     

       android中对数据操作包含有: file, sqlite3, Preferences, ContectResolver与ContentProvider前三种数据操作方式都只是针对本应用内数据,程序不能通过这三种方法去操作别的应用内的数据。 android中提供ContectResolver与ContentProvider来操作别的应用程序的数据。 一、 使用方式 一个应用实现ContentProvider来提供内容给别的应用来操作, 一个应用通过ContentResolver来操作别的应用数据,当然在自己的应用中也可以。 1. ContentResolver的获取    通过Context类:   

    Java代码 public abstract ContentResolver getContentResolver();   public abstract ContentResolver getContentResolver();

       2. ContentResolver常用操作   

    Java代码 //查询:    public final Cursor query(Uri uri, String[] projection,               String selection, String[] selectionArgs, String sortOrder);    //新增    public final Uri insert(Uri url, ContentValues values)        //更新    public final int update(Uri uri, ContentValues values, String where,                 String[] selectionArgs)    //删除    public final int delete(Uri url, String where, String[] selectionArgs)              //查询: public final Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder); //新增 public final Uri insert(Uri url, ContentValues values) //更新 public final int update(Uri uri, ContentValues values, String where, String[] selectionArgs) //删除 public final int delete(Uri url, String where, String[] selectionArgs)

           以上操作实际是通过Uri来匹配ContentProvider, 再由ContentProvider来进行具体操作的。        操作的参数和操作sqlite各函数的参数意义是一样的。        二、实现ContentProvider提供给外界访问 调用者ContentResoler是通过一个Uri来找到相应的ContentProvider的来进行实际操作。      1. Uri概念         一个Uri的样子如:      

    Java代码 scheme://authorities/path/id   scheme://authorities/path/id

           如电话记录:       

    Java代码 public static final Uri CONTENT_URI = Uri.parse("content://call_log/calls");   public static final Uri CONTENT_URI = Uri.parse("content://call_log/calls");

           a.根据scheme不同调用不程序来处理, 常用的:content, android_resource, file, http等        b.authorities是provider定义的,在AndroidManifest.xml中定义        c.path和id就好理解的。           2. Uri定义        创建自己的Uri, 如:       

    Java代码 content://com.shguo.statistic/sms   content://com.shguo.statistic/sms

           一般数据中都有dir和item两种(当然可定义多个)。为ContentProvider创建息的UriMatcher并添加这两者:       

    Java代码 String AUTHORITY = "com.shguo.statistics";    UriMatcher sUriMatcher = new UriMatcher(UriMatcher.NO_MATCH);    sUriMatcher.addURI(AUTHORITY, "sms",   SMS_DIR);   //SMS_DIR = 1    sUriMatcher.addURI(AUTHORITY, "sms/#", SMS_ITEM); //SMS_ITEM = 2   String AUTHORITY = "com.shguo.statistics"; UriMatcher sUriMatcher = new UriMatcher(UriMatcher.NO_MATCH); sUriMatcher.addURI(AUTHORITY, "sms", SMS_DIR); //SMS_DIR = 1 sUriMatcher.addURI(AUTHORITY, "sms/#", SMS_ITEM); //SMS_ITEM = 2

           contentProvider要根据传入uri判断是dir还是item来操作的。       

    Java代码 switch (sUriMatcher.match(uri))    switch (sUriMatcher.match(uri))

           来分步操作.             3. 定义MIME类型,       覆盖getType方法:主要是根据uri来返回Provider的MIME类型     

    Java代码 public static final String CONTENT_TYPE = "vnd.android.cursor.dir/vnd.shguo.sms";    ublic static final String CONTENT_ITEM_TYPE = "vnd.android.cursor.item/vnd.shguo.sms";   public static final String CONTENT_TYPE = "vnd.android.cursor.dir/vnd.shguo.sms"; public static final String CONTENT_ITEM_TYPE = "vnd.android.cursor.item/vnd.shguo.sms";

          getType()为:      

    Java代码 switch (sUriMatcher.match(uri)) {            case SMS_DIR:                return   CONTENT_TYPE;            case SMS_ITEM:                return CONTENT_ITEM_TYPE;            default:                throw new IllegalArgumentException("Unknown URI " + uri);         }   switch (sUriMatcher.match(uri)) { case SMS_DIR: return CONTENT_TYPE; case SMS_ITEM: return CONTENT_ITEM_TYPE; default: throw new IllegalArgumentException("Unknown URI " + uri); }

              4. 实现query, insert, delete, update四个操作。        具体的实现可以用sqlite, file等。并根据uri分情况操作。        a. query时如果是item加查询条件id.           where = "_ID=" + uri.getPathSegments().get(1)   + (!TextUtils.isEmpty(where) ? " AND (" + where + ')' : "";           最后要加上          cursor.setNotificationUri(getContext().getContentResolver(), uri);                b. insert时要求uri只能是dir. 成功之后返回一个加id的uri.          

    Java代码 Uri insertUri = ContentUris.withAppendedId(CONTENT_URI, rowId);   Uri insertUri = ContentUris.withAppendedId(CONTENT_URI, rowId);

           c. update、delete与query差不多。          

    Java代码 //注意通知注册uri的观察者。    getContext().getContentResolver().notifyChange(uri, null);   //注意通知注册uri的观察者。 getContext().getContentResolver().notifyChange(uri, null);

                           5. 在AndroidManifest.xml中定义         provider元素,主要属性有:       

    Java代码 name => ContentProvider类名    authorities => content type的授权部分    multiprocess => true允许在每个客户进程中创建provider实例,消除执行IPC的需求。

     

    ----------------------------------------------------------------------------------------------------------------------------------------------------

     

    android有一个独特之处就是,数据库只能被它的创建者所使用,其他的应用是不能访问到的,所以如果你想实现不同应用之间的数据共享,就不得不用content provider了。   在Android中,content provider是一个特殊的存储数据的类型,它提供了一套标准的接口用来获取以及操作数据。并且,android自身也提供了几个现成的content provider:Contacts, Browser, CallLog, Settings, MediaStore.       应用可以通过一个唯一的ContentResolver interface来使用具体的某个content provider。       ContentResolver cr = getContentResolver();   然后你就可以用ContentResolver提供的方法来使用你需要的content provider了。其中contentResolver提供的方法包括query(),insert(),update()等。要使用这些方法,还会涉及到一个东西,那就是Uri。你可以将它理解成一个string形式的contentProvider的完全路径,它的形式为<standard_prefix>://<authority>/<data_path>/<id>,   例如:   content://browser/bookmarks   content://contacts/people   content://contacts/people/3   下面结合一个实例来看我们如何使用一个已有的content provider,给例子展示了如何从已有的电话本中读出联系人信息:     package com.android.cp;     import android.app.Activity;   import android.content.ContentResolver;   import android.database.Cursor;   import android.os.Bundle;   import android.provider.Contacts.People;   import android.util.Log;   import android.widget.Toast;     public class ContentProviderTest extends Activity {       private final String TAG = "ContentProviderTest";        /** Called when the activity is first created. */      @Override      public void onCreate(Bundle savedInstanceState) {           super.onCreate(savedInstanceState);           Log.i(TAG,"enter onCreate");           setContentView(R.layout.main);           createCP();       }              public void createCP()       {           ContentResolver cr = getContentResolver();                      //Cursor cur = managedQuery(People.CONTENT_URI, null, null, null, null);             Cursor cur = cr.query(People.CONTENT_URI, nullnullnullnull);                      getColumnData(cur);         }              private void getColumnData(Cursor cur){            if (cur.moveToFirst()) {                 String name;                String phoneNumber;                int nameColumn = cur.getColumnIndex(People.NAME);                int phoneColumn = cur.getColumnIndex(People.NUMBER);                          do {                   // Get the field values                     name = cur.getString(nameColumn);                   phoneNumber = cur.getString(phoneColumn);                                     Log.i(TAG, "name="+name);                   DisplayToast(name+" "+phoneNumber);                 } while (cur.moveToNext());             }       }              public void DisplayToast(String s)       {       Toast.makeText(this,       s,       Toast.LENGTH_LONG).show();       }     }   需要注意的是,你需要在你的Manifest文件中加上     <uses-permission    android:name="android.permission.READ_CONTACTS">   </uses-permission>   否则,程序无法成功运行。  


    最新回复(0)