国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 系統 > Android > 正文

android基礎總結篇之八:創建及調用自己的ContentProvider

2019-12-12 04:36:18
字體:
來源:轉載
供稿:網友

今天我們來講解一下如何創建及調用自己的ContentProvider。

在前面兩篇文章中我們分別講了如何讀寫聯系人和短消息,相信大家對于ContentProvider的操作方法已經有了一定程度的了解。在有些場合,除了操作ContentProvider之外,我們還有可能需要創建自己的ContentProvider,來提供信息共享的服務,這就要求我們很好的掌握ContentProvider的創建及使用技巧。下面我們就由表及里的逐步講解每個步驟。

在正式開始實例演示之前,我們先來了解以下兩個知識點:

授權:

在Android中,每一個ContentProvider都會用類似于域名的字符串來注冊自己,我們成為授權(authority)。這個唯一標識的字符串是此ContentProvider可提供的一組URI的基礎,有了這個基礎,才能夠向外界提供信息的共享服務。

授權是在AndroidManifest.xml中完成的,每一個ContentProvider必須在此聲明并授權,方式如下:

<provider android:name=".SomeProvider"   android:authorities="com.your-company.SomeProvider"/> 

上面的<provider>元素指明了ContentProvider的提供者是“SomeProvider”這個類,并為其授權,授權的基礎URI為“com.your-company.SomeProvider”。有了這個授權信息,系統可以準確的定位到具體的ContentProvider,從而使訪問者能夠獲取到指定的信息。這和瀏覽Web頁面的方式很相似,“SomeProvider”就像一臺具體的服務器,而“com.your-company.SomeProvider”就像注冊的域名,相信大家對這個概念并不陌生,由此聯想一下就可以了解ContentProvider授權的作用了。(需要注意的是,除了Android內置應用程序之外,第三方程序應盡量使用以上方式的完全限定的授權名。)

MIME類型:

就像網站返回給定URL的MIME(Multipurpose Internet Mail Extensions,多用途Internet郵件擴展)類型一樣(這使瀏覽器能夠用正確的程序來查看內容),ContentProvider還負責返回給定URI的MIME類型。根據MIME類型規范,MIME類型包含兩部分:類型和子類型。例如:text/html,text/css,text/xml等等。

Android也遵循類似的約定來定義MIME類型。

對于單條記錄,MIME類型類似于:

vnd.android.cursor.item/vnd.your-company.content-type

而對于記錄的集合,MIME類型類似于:

vnd.android.cursor.dir/vnd.your-company.comtent-type

其中的vnd表示這些類型和子類型具有非標準的、供應商特定的形式;content-type可以根據ContentProvider的功能來定,比如日記的ContentProvider可以為note,日程安排的ContentProvider可以為schedule,等等。

了解了以上兩個知識點之后,我們就結合實例來演示一下具體的過程。

我們將會創建一個記錄person信息的ContentProvider,實現對person的CRUD操作,訪問者可以通過下面路徑操作我們的ContentProvider:

訪問者可以通過“[BASE_URI]/persons”來操作person集合,也可以通過“[BASE_URI]/persons/#”的形式操作單個person。
我們創建一個person的ContentProvider需要兩個步驟:

1.創建PersonProvider類:

我們需要繼承ContentProvider類,實現onCreate、query、insert、update、delete和getType這幾個方法。具體代碼如下:

package com.scott.provider;  import android.content.ContentProvider; import android.content.ContentUris; import android.content.ContentValues; import android.content.UriMatcher; import android.database.Cursor; import android.database.sqlite.SQLiteDatabase; import android.net.Uri;  public class PersonProvider extends ContentProvider {    private static final UriMatcher matcher;   private DBHelper helper;   private SQLiteDatabase db;      private static final String AUTHORITY = "com.scott.provider.PersonProvider";   private static final int PERSON_ALL = 0;   private static final int PERSON_ONE = 1;      public static final String CONTENT_TYPE = "vnd.android.cursor.dir/vnd.scott.person";   public static final String CONTENT_ITEM_TYPE = "vnd.android.cursor.item/vnd.scott.person";      //數據改變后立即重新查詢   private static final Uri NOTIFY_URI = Uri.parse("content://" + AUTHORITY + "/persons");      static {     matcher = new UriMatcher(UriMatcher.NO_MATCH);          matcher.addURI(AUTHORITY, "persons", PERSON_ALL);  //匹配記錄集合     matcher.addURI(AUTHORITY, "persons/#", PERSON_ONE); //匹配單條記錄   }      @Override   public boolean onCreate() {     helper = new DBHelper(getContext());     return true;   }    @Override   public String getType(Uri uri) {     int match = matcher.match(uri);     switch (match) {     case PERSON_ALL:       return CONTENT_TYPE;     case PERSON_ONE:       return CONTENT_ITEM_TYPE;     default:       throw new IllegalArgumentException("Unknown URI: " + uri);     }   }      @Override   public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {     db = helper.getReadableDatabase();     int match = matcher.match(uri);     switch (match) {     case PERSON_ALL:       //doesn't need any code in my provider.       break;     case PERSON_ONE:       long _id = ContentUris.parseId(uri);       selection = "_id = ?";       selectionArgs = new String[]{String.valueOf(_id)};       break;     default:       throw new IllegalArgumentException("Unknown URI: " + uri);     }     return db.query("person", projection, selection, selectionArgs, null, null, sortOrder);   }    @Override   public Uri insert(Uri uri, ContentValues values) {     int match = matcher.match(uri);     if (match != PERSON_ALL) {       throw new IllegalArgumentException("Wrong URI: " + uri);     }     db = helper.getWritableDatabase();     if (values == null) {       values = new ContentValues();       values.put("name", "no name");       values.put("age", "1");       values.put("info", "no info.");     }     long rowId = db.insert("person", null, values);     if (rowId > 0) {       notifyDataChanged();       return ContentUris.withAppendedId(uri, rowId);     }     return null;   }    @Override   public int delete(Uri uri, String selection, String[] selectionArgs) {     db = helper.getWritableDatabase();     int match = matcher.match(uri);     switch (match) {     case PERSON_ALL:       //doesn't need any code in my provider.       break;     case PERSON_ONE:       long _id = ContentUris.parseId(uri);       selection = "_id = ?";       selectionArgs = new String[]{String.valueOf(_id)};     }     int count = db.delete("person", selection, selectionArgs);     if (count > 0) {       notifyDataChanged();     }     return count;   }    @Override   public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {     db = helper.getWritableDatabase();     int match = matcher.match(uri);     switch (match) {     case PERSON_ALL:       //doesn't need any code in my provider.       break;     case PERSON_ONE:       long _id = ContentUris.parseId(uri);       selection = "_id = ?";       selectionArgs = new String[]{String.valueOf(_id)};       break;     default:       throw new IllegalArgumentException("Unknown URI: " + uri);     }     int count = db.update("person", values, selection, selectionArgs);     if (count > 0) {       notifyDataChanged();     }     return count;   }    //通知指定URI數據已改變   private void notifyDataChanged() {     getContext().getContentResolver().notifyChange(NOTIFY_URI, null);       } } 

在PersonProvider中,我們定義了授權地址為“com.scott.provider.PersonProvider”,相信大家在前面也有所了解了?;谶@個授權,我們使用了一個UriMatcher對其路徑進行匹配,“[BASE_URI]/persons"和“[BASE_URI]/persons/#”這兩種路徑我們在上面也介紹過,分別對應記錄集合和單個記錄的操作。在query、insert、update和delete方法中我們根據UriMatcher匹配結果來判斷該URI是操作記錄集合還是單條記錄,從而采取不同的處理方法。在getType方法中,我們會根據匹配的結果返回不同的MIME類型,這一步是不能缺少的,比如我們在query方法中有可能是查詢全部集合,有可能是查詢單條記錄,那么我們返回的Cursor或是集合類型,或是單條記錄,這個跟getType返回的MIME類型是一致的,就好像瀏覽網頁一樣,指定的url返回的信息是什么類型,那么瀏覽器就應該接收到對應的MIME類型。另外,我們注意到,上面代碼中,在insert、update、delete方法中都調用了notifyDataChanged方法,這個方法中僅有的一步操作就是通知“[BASE_URI]/persons"的訪問者,數據發生改變了,應該重新加載了。

在我們的PersonProvider中,我們用到了Person、DBHelper類,代碼如下:

package com.scott.provider;  public class Person {   public int _id;   public String name;   public int age;   public String info;      public Person() {   }      public Person(String name, int age, String info) {     this.name = name;     this.age = age;     this.info = info;   } } [java] view plain copypackage com.scott.provider;  import android.content.Context; import android.database.sqlite.SQLiteDatabase; import android.database.sqlite.SQLiteOpenHelper;  public class DBHelper extends SQLiteOpenHelper {    private static final String DATABASE_NAME = "provider.db";   private static final int DATABASE_VERSION = 1;      public DBHelper(Context context) {     super(context, DATABASE_NAME, null, DATABASE_VERSION);   }    @Override   public void onCreate(SQLiteDatabase db) {     String sql = "CREATE TABLE IF NOT EXISTS person" +         "(_id INTEGER PRIMARY KEY AUTOINCREMENT, name VARCHAR, age INTEGER, info TEXT)";     db.execSQL(sql);   }    @Override   public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {     db.execSQL("DROP TABLE IF EXISTS person");     onCreate(db);   } } 

最后,要想讓這個ContentProvider生效,我們需要在AndroidManifest.xml中聲明并為其授權,如下所示:

<provider android:name=".PersonProvider"   android:authorities="com.scott.provider.PersonProvider"   android:multiprocess="true"/> 

其中,android:multiprocess代表是否允許多進程操作。另外我們也可以為其聲明相應的權限,對應的屬性是:android:permission。

2.調用PersonProvider類:

完成了person的ContentProvider后,下面我們來看一下如何訪問它。這一步我們在MainActivity中完成,看下面代碼:

package com.scott.provider;  import java.util.ArrayList;  import android.app.Activity; import android.content.ContentResolver; import android.content.ContentUris; import android.content.ContentValues; import android.database.Cursor; import android.database.CursorWrapper; import android.net.Uri; import android.os.Bundle; import android.os.Handler; import android.os.Message; import android.view.View; import android.widget.ListView; import android.widget.SimpleCursorAdapter;   public class MainActivity extends Activity {      private ContentResolver resolver;   private ListView listView;      private static final String AUTHORITY = "com.scott.provider.PersonProvider";   private static final Uri PERSON_ALL_URI = Uri.parse("content://" + AUTHORITY + "/persons");      private Handler handler = new Handler() {     public void handleMessage(Message msg) {       //update records.       requery();     };   };      @Override   public void onCreate(Bundle savedInstanceState) {     super.onCreate(savedInstanceState);     setContentView(R.layout.main);      resolver = getContentResolver();     listView = (ListView) findViewById(R.id.listView);          //為PERSON_ALL_URI注冊變化通知     getContentResolver().registerContentObserver(PERSON_ALL_URI, true, new PersonObserver(handler));   }      /**    * 初始化    * @param view    */   public void init(View view) {     ArrayList<Person> persons = new ArrayList<Person>();          Person person1 = new Person("Ella", 22, "lively girl");     Person person2 = new Person("Jenny", 22, "beautiful girl");     Person person3 = new Person("Jessica", 23, "sexy girl");     Person person4 = new Person("Kelly", 23, "hot baby");     Person person5 = new Person("Jane", 25, "pretty woman");          persons.add(person1);     persons.add(person2);     persons.add(person3);     persons.add(person4);     persons.add(person5);      for (Person person : persons) {       ContentValues values = new ContentValues();       values.put("name", person.name);       values.put("age", person.age);       values.put("info", person.info);       resolver.insert(PERSON_ALL_URI, values);     }   }      /**    * 查詢所有記錄    * @param view    */   public void query(View view) { //   Uri personOneUri = ContentUris.withAppendedId(PERSON_ALL_URI, 1);查詢_id為1的記錄     Cursor c = resolver.query(PERSON_ALL_URI, null, null, null, null);          CursorWrapper cursorWrapper = new CursorWrapper(c) {              @Override       public String getString(int columnIndex) {         //將簡介前加上年齡         if (getColumnName(columnIndex).equals("info")) {           int age = getInt(getColumnIndex("age"));           return age + " years old, " + super.getString(columnIndex);         }         return super.getString(columnIndex);       }     };          //Cursor須含有"_id"字段     SimpleCursorAdapter adapter = new SimpleCursorAdapter(this, android.R.layout.simple_list_item_2,         cursorWrapper, new String[]{"name", "info"}, new int[]{android.R.id.text1, android.R.id.text2});     listView.setAdapter(adapter);          startManagingCursor(cursorWrapper); //管理Cursor   }      /**    * 插入一條記錄    * @param view    */   public void insert(View view) {     Person person = new Person("Alina", 26, "attractive lady");     ContentValues values = new ContentValues();     values.put("name", person.name);     values.put("age", person.age);     values.put("info", person.info);     resolver.insert(PERSON_ALL_URI, values);   }      /**    * 更新一條記錄    * @param view    */   public void update(View view) {     Person person = new Person();     person.name = "Jane";     person.age = 30;     //將指定name的記錄age字段更新為30     ContentValues values = new ContentValues();     values.put("age", person.age);     resolver.update(PERSON_ALL_URI, values, "name = ?", new String[]{person.name});          //將_id為1的age更新為30 //   Uri updateUri = ContentUris.withAppendedId(PERSON_ALL_URI, 1); //   resolver.update(updateUri, values, null, null);   }      /**    * 刪除一條記錄    * @param view    */   public void delete(View view) {     //刪除_id為1的記錄     Uri delUri = ContentUris.withAppendedId(PERSON_ALL_URI, 1);     resolver.delete(delUri, null, null);          //刪除所有記錄 //   resolver.delete(PERSON_ALL_URI, null, null);   }      /**    * 重新查詢    */   private void requery() {     //實際操作中可以查詢集合信息后Adapter.notifyDataSetChanged();     query(null);   } } 

我們看到,在上面的代碼中,分別對應每一種情況進行測試,相對較為簡單。我們主要講一下registerContentObserver這一環節。

在前面的PersonProvider我們也提到,在數據更改后,會向指定的URI訪問者發出通知,以便于更新查詢記錄。大家注意,僅僅是ContentProvider出力還不夠,我們還需要在訪問者中注冊一個ContentObserver,才能夠接收到這個通知。下面我們創建一個

PersonObserver:package com.scott.provider;  import android.database.ContentObserver; import android.os.Handler; import android.os.Message; import android.util.Log;  public class PersonObserver extends ContentObserver {    public static final String TAG = "PersonObserver";   private Handler handler;      public PersonObserver(Handler handler) {     super(handler);     this.handler = handler;   }      @Override   public void onChange(boolean selfChange) {     super.onChange(selfChange);     Log.i(TAG, "data changed, try to requery.");     //向handler發送消息,更新查詢記錄     Message msg = new Message();     handler.sendMessage(msg);   } } 

這樣一來,當ContentProvider發來通知之后,我們就能立即接收到,從而向handler發送一條消息,重新查詢記錄,使我們能夠看到最新的記錄信息。

最后,我們要在AndroidManifest.xml中為MainActivity添加MIME類型過濾器,告訴系統MainActivity可以處理的信息類型:

<!-- MIME類型 --> <intent-filter>   <data android:mimeType="vnd.android.cursor.dir/vnd.scott.person"/> </intent-filter> <intent-filter>   <data android:mimeType="vnd.android.cursor.item/vnd.scott.person"/> </intent-filter> 

這樣就完成了訪問者的代碼,我們來看一下效果:

鑒于操作類型太多,我在這里就不再展示了,大家可以自己試一試。

原文鏈接:http://blog.csdn.net/liuhe688/article/details/7050868

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持武林網。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 德州市| 抚顺市| 东兰县| 信阳市| 景德镇市| 秀山| 临沧市| 三原县| 霸州市| 噶尔县| 邓州市| 井冈山市| 大石桥市| 绵竹市| 汉寿县| 蓬溪县| 潼南县| 武威市| 太康县| 河池市| 四子王旗| 宁南县| 凤凰县| 长宁县| 怀仁县| 陈巴尔虎旗| 泌阳县| 富蕴县| 松滋市| 新源县| 石狮市| 禄丰县| 盱眙县| 辽宁省| 泰和县| 准格尔旗| 库尔勒市| 迭部县| 枣阳市| 苍梧县| 廊坊市|