ContentProvider概述
public abstract class
ContentProvider
extends Object
implements ComponentCallbacks2
Content providers are one of the primary building blocks of Android applications, providing content to applications. They encapsulate data and provide it to applications through the single ContentResolver interface. A content provider is only required if you need to share data between multiple applications. For example, the contacts data is used by multiple applications and must be stored in a content provider. If you don’t need to share data amongst multiple applications you can use a database directly via SQLiteDatabase.
When a request is made via a ContentResolver the system inspects the authority of the given URI and passes the request to the content provider registered with the authority. The content provider can interpret the rest of the URI however it wants. The UriMatcher class is helpful for parsing URIs.
Android的ContentProvider作用就是在应用程序之间交换数据,如果我们采用文件共享的方式需要注意的是必须采用特定的数据访问方式,而ContentProvider是不同应用程序之间进行数据交换的标准API。ContentProvider以某种Uri的形式对外提供数据,而其他应用使用ContentResolver根据Uri去访问操作指定数据。
ContentProvider也是Android应用的四大组件之一,与Activity、Service、BroadcastReceiver相似,都需要在AndroidManifest.xml文件中进行配置。
ContentProvider的使用
- 定义自己的ContentProvider类,继承子ContentProvider基类;
- 在AndroidManifest.xml文件中注册,并且绑定一个URL。
public class ImageContentProvider extends ContentProvider{
//该方法用于供外部应用从ContentProvider删除数据 @Override public int delete(Uri uri, String selection, String[] selectionArgs) { return 0; } //该方法用于返回当前Url所代表数据的MIME类型 @Override public String getType(Uri uri) { return null; } //该方法用于供外部应用往ContentProvider添加数据 @Override public Uri insert(Uri uri, ContentValues values) { return null; } //该方法在ContentProvider创建后就会被调用,Android开机后,ContentProvider在其它应用第一次访问它时才会被创建 @Override public boolean onCreate() { return false; } //该方法用于供外部应用从ContentProvider中获取数据 @Override public Cursor query(Uri uri, String[] values, String selection, String[] selectionArgs, String sortOrder) { return null; } //该方法用于供外部应用更新ContentProvider中的数据 @Override public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) { return 0; }
}
如果操作的数据属于集合类型,那么MIME类型字符串应该以vnd.android.cursor.dir/开头,
例如:要得到所有person记录的Uri为content://com.ljq.provider.personprovider/person,那么返回的MIME类型字符串应该为:”vnd.android.cursor.dir/person”。
如果要操作的数据属于非集合类型数据,那么MIME类型字符串应该以vnd.android.cursor.item/开头,
例如:得到id为10的person记录,Uri为content://com.ljq.provider.personprovider/person/10,那么返回的MIME类型字符串为:”vnd.android.cursor.item/person”。
注册ContentProvider只需要在
Uri概述
Uri还是很容易理解的,我们可以和互联网的URL进行类比。
content://com.twilight.providers.imagecontentprovider/image/2
这里的意思就是访问image数据总ID为2的记录。虽然大部分ContentProvider操作的数据都是来自于数据库,但是有时候可以来自于文件、XML或网络等其他的存储方式。
content://com.twilight.providers.imagecontentprovider/image/detail
这里的意思就是操作image节点下的detail节点。
Uri工具类
Android系统提供了2个操作Uri的工具类,分别是UriMatcher和ContentUris。
(1)UriMatcher类是用来匹配Uri
UriMatcher matcher = new UriMatcher(UriMatcher.NO_MATCH);
matcher.addURI("com.twilight.providers.imageprovider", "image", 1);
matcher.addURI("com.twilight.providers.imageprovider", "image/#", 2);
switch (matcher.match(Uri.parse("content://com.twilight.providers.imageprovider/image/7"))) {
case 1:
break;
case 2:
break;
default:
break;
}
注册完需要匹配的Uri后,是用matcher.match(uri)方法对输入的Uri进行匹配,如果匹配就返回匹配码。
(2)ConentUris类用于操作Uri路径后面的ID部分。
Uri uri1 = Uri.parse("content://com.twilight.providers.imageprovider/image");
Uri resultUri = ContentUris.withAppendedId(uri1, 7);
Uri uri2 = Uri.parse("content://com.twilight.providers.imageprovider/image/7");
long resultId = ContentUris.parseId(uri2);
使用ContentResolver操作数据
ContentProvider相当于一个网站,而ContentResolver则相对于HttpClient。一般来说Content是单例模式,当多个应用程序通过ContentResolver来操作ContentProvider提供的数据时,数据操作会委托给同一个ContentProvider处理。
(1)调用Activity的ContentResolver()获取ContentResolver对象;
(2)使用ContentResolover的insert、delete、update、query方法操作数据,如果是系统提供的ContentProvider,只需要知道ContentProvider的Uri即可。
ContentResolver resolver = getContentResolver();
Uri uri = Uri.parse("content://com.twilight.providers.imageprovider/image");
ContentValues values = new ContentValues();
values.put("name", "pic1");
values.put("date", "2014-03-10");
resolver.insert(uri, values);
监听ContentProvider的数据改变
getContext().getContentResolver().notifyChange(uri, null);
当ContentProvider所共享的数据发生了改变,这行代码可以通知所有注册在改Uri上的监听者,为了在应用程序中监听ContentProvider数据的改变,需要利用Android提供了ContentObserver基类。
监听ContentProvider数据改变的监听器需要继承ContentObserver类,并重写改基类所定义的onChange(boolean selfChange)方法。在使用的时候需要通过ContentProvider向指定的Uri注册ContentObserver监听器。
final class ImageObserver extends ContentObserver {
public ImageObserver(Handler handler) {
super(handler);
}
public void onChange(boolean selfChange) {
}
}
getContentResolver().registerContentObserver(Uri.parse("content://com.twilight.providers.imageprovider/image"),
true, new ImageObserver(new Handler()));