ContentProvider能夠將應用中的數據共享給其餘應用訪問,其餘應用能夠經過ContentProvider 對你應用中的數據進行添刪改查。好處在於:統一了數據的訪問方式。java
如何使用ContentProvider呢?android
1、繼承ContentProvider並重寫下面方法:
public class PersonContentProvider extends ContentProvider{
public boolean onCreate() 能夠進行數據化的初始化操做
public Uri insert(Uri uri, ContentValues values) 向ContentProvider中插入數據
public int delete(Uri uri, String selection, String[] selectionArgs) 刪除ContentProvider中數據庫
更新ContentProvider中的數據
網絡
public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs)app
//能夠供外部應用查詢ContentProvider中的數據
public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder)
public String getType(Uri uri)返回操做數據的內容類型ide
}工具
該方法用於返回當前Url所表明數據的MIME類型。若是操做的數據屬於集合類型,那麼MIME類型字符串應該以 vnd.android.cursor.dir/開頭,例如:要獲得全部person記錄的Uri爲content://cn.itcast.provider.personprovider/person,那麼返回的MIME類型字符串應該爲:「vnd.android.cursor.dir/person」。若是要操做的數據屬於非集合類型數據,那麼MIME類型字符串應該以vnd.android.cursor.item/開頭,例如:獲得id爲10的person記錄,Uri爲content://cn.itcast.provider.personprovider/person/10,那麼返回的MIME類型字符串應該爲:「vnd.android.cursor.item/person」。
網站
2、須要在AndroidManifest.xml使用<provider>對該ContentProvider進行配置,爲了能讓其餘應用找到該 ContentProvider , ContentProvider 採用了authorities(主機名/域名)對它進行惟一標識,你能夠把 ContentProvider看做是一個網站(想一想,網站也是提供數據者),authorities 就是他的域名:
spa
<manifest .... > <application android:icon="@drawable/icon" android:label="@string/app_name"> <provider android:name=".PersonContentProvider" android:authorities="cn.meng.providers.personprovider"/> </application> </manifest>
咱們看到了重寫方法中有Uri。那麼咱們就介紹下Uri。code
Uri
Uri表明了要操做的數據,Uri主要包含了兩部分信息:1》須要操做的ContentProvider ,2》對ContentProvider中的什麼數據進行操做,一個Uri由如下幾部分組成:
content://cn.meng.providers.personprovider/person/10
表示要操做person表中id爲10的記錄。
ContentProvider(內容提供者)的scheme已經由Android所規定, scheme爲:content://
主機名(或叫Authority)用於惟一標識這個ContentProvider,外部調用者能夠根據這個標識來找到它。
路徑(path)能夠用來表示咱們要操做的數據,路徑的構建應根據業務而定,以下:
要操做person表中id爲10的記錄,能夠構建這樣的路徑:/person/10
要操做person表中id爲10的記錄的name字段, person/10/name
要操做person表中的全部記錄,能夠構建這樣的路徑:/person
要操做xxx表中的記錄,能夠構建這樣的路徑:/xxx
固然要操做的數據不必定來自數據庫,也能夠是文件、xml或網絡等其餘存儲方式,以下:
要操做xml文件中person節點下的name節點,能夠構建這樣的路徑:/person/name
若是要把一個字符串轉換成Uri,可使用Uri類中的parse()方法,以下:
Uri uri = Uri.parse("content://cn.itcast.provider.personprovider/person")
UriMatcher
由於Uri表明了要操做的數據,因此咱們常常須要解析Uri,並從Uri中獲取數據。Android系統提供了兩個用於操做Uri的工具類,分別爲UriMatcher 和ContentUris 。掌握它們的使用,會便於咱們的開發工做。
UriMatcher類用於匹配Uri,它的用法以下:
一、首先第一步把你須要匹配Uri路徑所有給註冊上,以下:
//常量UriMatcher.NO_MATCH表示不匹配任何路徑的返回碼 UriMatcher sMatcher = new UriMatcher(UriMatcher.NO_MATCH); //若是match()方法匹配content://cn.itcast.provider.personprovider/person路徑,返回匹配碼爲1 sMatcher.addURI(「cn.itcast.provider.personprovider」, 「person」, 1); //添加須要匹配uri,若是匹配就會返回匹配碼 //若是match()方法匹配content://cn.itcast.provider.personprovider/person/230路徑,返回匹配碼爲2 sMatcher.addURI(「cn.itcast.provider.personprovider」, 「person/#」, 2);//#號爲通配符 switch (sMatcher.match(Uri.parse("content://cn.itcast.provider.personprovider/person/10"))) { case 1 break; case 2 break; default://不匹配 break; }
註冊完須要匹配的Uri後,就可使用sMatcher.match(uri)方法對輸入的Uri進行匹配,若是匹配就返回匹配碼,匹配碼是調用addURI()方法傳入的第三個參數,假設匹配content://cn.itcast.provider.personprovider/person路徑,返回的匹配碼爲1
ContentUris
ContentUris類用於獲取Uri路徑後面的ID部分,它有兩個比較實用的方法:
一、withAppendedId(uri, id)用於爲路徑加上ID部分:
Uri uri = Uri.parse("content://cn.itcast.provider.personprovider/person")
Uri resultUri = ContentUris.withAppendedId(uri, 10);
//生成後的Uri爲:content://cn.itcast.provider.personprovider/person/10
二、parseId(uri)方法用於從路徑中獲取ID部分:
Uri uri = Uri.parse("content://cn.itcast.provider.personprovider/person/10")
long personid = ContentUris.parseId(uri);//獲取的結果爲:10
ContentResolver
當外部應用須要對ContentProvider中的數據進行添加、刪除、修改和查詢操做時,可使用ContentResolver 類來完成,要獲取ContentResolver 對象,可使用Activity提供的getContentResolver()方法。 ContentResolver 類提供了與 ContentProvider類相同簽名的四個方法:
public Uri insert(Uri uri, ContentValues values)
該方法用於往ContentProvider添加數據。
public int delete(Uri uri, String selection, String[] selectionArgs)
該方法用於從ContentProvider刪除數據。
public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs)
該方法用於更新ContentProvider中的數據。
public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder)
該方法用於從ContentProvider中獲取數據。
這些方法的第一個參數爲Uri,表明要操做的ContentProvider和對其中的什麼數據進行操做,假設給定的是: Uri.parse(「content://cn.itcast.providers.personprovider/person/10」),那麼將會對主機名爲 cn.itcast.providers.personprovider的ContentProvider進行操做,操做的數據爲person表中id爲10的記錄。
使用ContentResolver對ContentProvider中的數據進行添加、刪除、修改和查詢操做:
ContentResolver resolver = getContentResolver(); Uri uri = Uri.parse("content://cn.itcast.provider.personprovider/person"); //添加一條記錄 ContentValues values = new ContentValues(); values.put("name", "itcast"); values.put("age", 25); resolver.insert(uri, values); //獲取person表中全部記錄 Cursor cursor = resolver.query(uri, null, null, null, "personid desc"); while(cursor.moveToNext()){ Log.i("ContentTest", "personid="+ cursor.getInt(0)+ ",name="+ cursor.getString(1)); } //把id爲1的記錄的name字段值更改新爲liming ContentValues updateValues = new ContentValues(); updateValues.put("name", "liming"); Uri updateIdUri = ContentUris.withAppendedId(uri, 2); resolver.update(updateIdUri, updateValues, null, null); //刪除id爲2的記錄 Uri deleteIdUri = ContentUris.withAppendedId(uri, 2); resolver.delete(deleteIdUri, null, null);
監聽ContentProvider數據的變化
若是ContentProvider的訪問者須要知道ContentProvider中的數據發生變化,能夠在ContentProvider 發生數據變化時調 用getContentResolver().notifyChange(uri, null)來通知註冊在此URI上的訪問者,例子以下:
public class PersonContentProvider extends ContentProvider { public Uri insert(Uri uri, ContentValues values) { db.insert("person", "personid", values); getContext().getContentResolver().notifyChange(uri, null); } }
若是ContentProvider的訪問者須要獲得數據變化通知,必須使用ContentObserver對數據(數據採用uri描述)進行監聽,當監聽到數據變化通知時,系統就會調用ContentObserver的onChange()方法:
getContentResolver().registerContentObserver( Uri.parse("content://cn.itcast.providers.personprovider/person"),true, //true表示只要發出通知的Uri以方法第一個參數開頭都能被監聽到, //否側監聽uri必須與發出通知的uri徹底匹配才能被監聽到 new PersonObserver(new Handler())); public class PersonObserver extends ContentObserver{ public PersonObserver(Handler handler) { super(handler); } public void onChange(boolean selfChange) { //此處能夠進行相應的業務處理 } }