android之ContentResolver與ContentProvider

轉自:  http://hi.baidu.com/kaisep/blog/item/499d6d8ba682f41cc9fc7a49.html

android中對數據操作包含有: 
file, sqlite3, Preferences, ContectResolver與ContentProvider前三種數據操作方式都只是針對本應用內數據,程序不能通過這三種方法去操作別的應用內的數據。 
android中提供ContectResolver與ContentProvider來操作別的應用程序的數據。 

一、 使用方式 
一個應用實現ContentProvider來提供內容給別的應用來操作, 
一個應用通過ContentResolver來操作別的應用數據,當然在自己的應用中也可以。 
1. ContentResolver的獲取 
   通過Context類: 
  

  1. public abstract ContentResolver getContentResolver();  


   
2. ContentResolver常用操作 
  

  1. //查詢:   
  2. public final Cursor query(Uri uri, String[] projection,   
  3.            String selection, String[] selectionArgs, String sortOrder);   
  4. //新增   
  5. public final Uri insert(Uri url, ContentValues values)       
  6. //更新   
  7. public final int update(Uri uri, ContentValues values, String where,   
  8.              String[] selectionArgs)   
  9. //刪除   
  10. public final int delete(Uri url, String where, String[] selectionArgs)   
  11.          


       以上操作實際是通過Uri來匹配ContentProvider, 再由ContentProvider來進行具體操作的。 
       操作的參數和操作sqlite各函數的參數意義是一樣的。 
       
二、實現ContentProvider提供給外界訪問 
調用者ContentResoler是通過一個Uri來找到相應的ContentProvider的來進行實際操作。 
     1. Uri概念 
        一個Uri的樣子如: 
     

  1. scheme://authorities/path/id  


       如電話記錄: 
      

  1. 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, 如: 
      

  1. content://com.shguo.statistic/sms  


       一般數據中都有dir和item兩種(當然可定義多個)。爲ContentProvider創建息的UriMatcher並添加這兩者: 
      

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


       contentProvider要根據傳入uri判斷是dir還是item來操作的。 
      

  1. switch (sUriMatcher.match(uri))   


       來分步操作. 
       
     3. 定義MIME類型, 
      覆蓋getType方法:主要是根據uri來返回Provider的MIME類型 
    

  1. public static final String CONTENT_TYPE = "vnd.android.cursor.dir/vnd.shguo.sms";   
  2. ublic static final String CONTENT_ITEM_TYPE = "vnd.android.cursor.item/vnd.shguo.sms";  


      getType()爲: 
     

  1. switch (sUriMatcher.match(uri)) {   
  2.         case SMS_DIR:   
  3.             return   CONTENT_TYPE;   
  4.         case SMS_ITEM:   
  5.             return CONTENT_ITEM_TYPE;   
  6.         default:   
  7.             throw new IllegalArgumentException("Unknown URI " + uri);   
  8.      }  


     
     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. 
         

  1. Uri insertUri = ContentUris.withAppendedId(CONTENT_URI, rowId);  


       c. update、delete與query差不多。 
         

  1. //注意通知註冊uri的觀察者。   
  2. getContext().getContentResolver().notifyChange(uri, null);  


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

  1. name => ContentProvider類名   
  2. authorities => content type的授權部分   
  3. multiprocess => true允許在每個客戶進程中創建provider實例,消除執行IPC的需求。

 

發佈了90 篇原創文章 · 獲贊 7 · 訪問量 13萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章