Android中ContentProvider组件数据共享

ContentProvider的功能和意义:

主要用于对外共享数据,也就是通过ContentProvider把应用中的数据共享给其他应用访问,其他应用可以通过ContentProvider对指定应用中的数据进行操作。

实现在不同应用程序之间共享数据,在应用程序之间交换数据,一个应用程序A把自己的数据通过ContentProvider暴露给其他程序使用B,B就可以通过ContentResolver来操作ContentProvider暴露的数据,包括增,删,查,改

1、ContentProvider使用表的形式来组织数据
   无论数据的来源是什么,ContentProvider都会认为是一种表,然后把数据组织成表格
2、ContentProvider提供的方法
public boolean onCreate() 在创建ContentProvider时调用
public Cursor query(Uri, String[], String, String[], String) 用于查询指定Uri的ContentProvider,返回一个Cursor
public Uri insert(Uri, ContentValues) 用于添加数据到指定Uri的ContentProvider中
public int update(Uri, ContentValues, String, String[]) 用于更新指定Uri的ContentProvider中的数据
public int delete(Uri, String, String[]) 用于从指定Uri的ContentProvider中删除数据
public String getType(Uri) 用于返回指定的Uri中的数据的MIME类型
*如果操作的数据属于集合类型,那么MIME类型字符串应该以vnd.android.cursor.dir/开头。
例如:要得到所有person记录的Uri为content://contacts/person,那么返回的MIME类型字符串为"vnd.android.cursor.dir/person"。
*如果要操作的数据属于非集合类型数据,那么MIME类型字符串应该以vnd.android.cursor.item/开头。
例如:要得到id为10的person记录的Uri为content://contacts/person/10,那么返回的MIME类型字符串应为"vnd.android.cursor.item/person"。
3、每个ContentProvider都有一个公共的URI,这个URI用于表示这个ContentProvider所提供的数据。Android所提供的ContentProvider都存放在android.provider包当中


二,Uri类简介

1,为系统的每一个资源给其一个名字,比方说通话记录。每个ContentProvider都有一个公共的URI,这个URI用于表示这个ContentProvider所提供的数据。

例Uri:content://com.hust.uri/words

Uri指定了将要操作的ContentProvider,其实可以把一个Uri看作是一个网址,我们把Uri分为三部分。
第一部分是"content://"。可以看作是网址中的"http://"。
第二部分是主机名或authority,用于唯一标识这个ContentProvider,外部应用需要根据这个标识来找到它。可以看作是网址中的主机名,比如"blog.csdn.net"。
第三部分是路径名,资源部分,用来表示将要操作的数据。

例:content://com.hust.uri/words

访问的资源是words/2,意味着访问word数据表中的全部数据

例:content://com.hust.uri/words/2

访问的资源是words/2,意味着访问word数据中ID为2的记录

例:content://com.hust.uri/words/2/word

访问的资源是words/2,意味着访问word数据中ID为2的记录的word字段

把一个字符串转换成Uri,是Uri类的静态方法:

Uri uri = Uri.parse("content://com.hust.uri/contact")


2,UriMatcher类使用介绍

因为Uri代表了要操作的数据,所以我们经常需要解析Uri,并从Uri中获取数据。Android系统提供了两个用于操作Uri的工具类,分别为UriMatcher和ContentUris

UriMatcher类用于匹配Uri:

 A,void addURI(String authority,String path,int code)向UriMatcher中注册Uri,authority和path组成一个Uri,code代表该Uri对应的标识码

 B,int match(Uri uri);根据前面注册的Uri来判断指定的Uri对应的标识码

UriMatcher matcher=new UriMatcher(UriMatcher.NO_MATCH);
matcher.addURI("com.hust.uri","words",1); 
matcher.addURI("com.hust.uri","words/#",2); 
com.hust.uri/words/#表示words下的所有数据uri的表示为2

匹配结果如下:

matcher.match(Uri.parse("content://com.hust.uri/words"));
//返回标识码1
matcher.match(Uri.parse("content://com.hust.uri/words/2"));
//返回标识码2
matcher.match(Uri.parse("content://com.hust.uri/words/10"));
//返回标识码2
注册完需要匹配的Uri后,就可以使用sMatcher.match(uri)方法对输入的Uri进行匹配,如果匹配就返回匹配码,匹配码是调用addURI()方法传入的第三个参数

3,ContentUris类使用介绍

ontentUris类用于操作Uri路径后面的ID部分,它有两个比较实用的方法:
withAppendedId(uri, id)用于为路径加上ID部分:

Uri resultUri = ContentUris.withAppendedId(uri, 10); 
//生成后的Uri为:content://com.hust.personprovider/person/10
parseId(uri)方法用于从路径中获取ID部分:
Uri uri = Uri.parse("content://com.hust.uri.personprovider/person/10")
long personid = ContentUris.parseId(uri);//获取的结果为:10

三,开发ContentProvider

1,开发一个ContentProvider子类,继承ContentProvider,实现query,insert,update,delete等方法

2,在Manifest.xml清单文件中注册该ContentProvider,指定其Android:authorities属性

public class FirstProvider extends ContentProvider
{
	// 第一次创建该ContentProvider时调用该方法
	@Override
	public boolean onCreate()
	{
		System.out.println("===onCreate方法被调用===");
		return true;
	}

	// 该方法的返回值代表了该ContentProvider所提供数据的MIME类型
	@Override
	public String getType(Uri uri)
	{
		System.out.println("~~getType方法被调用~~");
		return null;
	}

	// 实现查询方法,该方法应该返回查询得到的Cursor
	@Override
	public Cursor query(Uri uri, String[] projection, String where,
		String[] whereArgs, String sortOrder)
	{
		System.out.println(uri + "===query方法被调用===");
		System.out.println("where参数为:" + where);
		return null;
	}

	// 实现插入的方法,该方法应该新插入的记录的Uri
	@Override
	public Uri insert(Uri uri, ContentValues values)
	{
		System.out.println(uri + "===insert方法被调用===");
		System.out.println("values参数为:" + values);
		return null;
	}

	// 实现删除方法,该方法应该返回被删除的记录条数
	@Override
	public int delete(Uri uri, String where, String[] whereArgs)
	{
		System.out.println(uri + "===delete方法被调用===");
		System.out.println("where参数为:" + where);
		return 0;
	}

	// 实现删除方法,该方法应该返回被更新的记录条数
	@Override
	public int update(Uri uri, ContentValues values, String where,
		String[] whereArgs)
	{
		System.out.println(uri + "===update方法被调用===");
		System.out.println("where参数为:"
			+ where + ",values参数为:" + values);
		return 0;
	}
}

这4个方法用于供其他应用通过ContentProvider调用。
注册ContentProvider:
<!-- 注册一个ContentProvider -->
		<provider
		        android:exported="true"
			android:name=".FirstProvider"
			android:authorities="org.providers.firstprovider">
		</provider>
android:authorities属性是一定要配置的,指定该ContentProvider对应的Uri

上面配置指定了该ContentProvider被绑定到“content://org.providers.firstprovider”,这意味着其他应用的ContentResovler向该Uri执行query,insert,update,delete方法时,实际上是调用该ContentProvider的query,insert,update,delete方法,ContentResovler调用方法时将参数传给ContentProvider对应的方法参数。


四,开发contentResovler

通过getContentResolver()方法获取ContentResolver对象,获取ContentResolver对象之后,接下来可调用query,insert,update,delete方法了,实际上调用的是指定Uri对应的ContentProvider的query,insert,update,delete方法

例:界面4个按钮,分别对应增删查改功能:

public class FirstResolver extends Activity
{
	ContentResolver contentResolver;//声明变量
	Uri uri = Uri.parse("content://org.providers.firstprovider/");//FirstProvider的Uri

	@Override
	public void onCreate(Bundle savedInstanceState)
	{
		super.onCreate(savedInstanceState);
		setContentView(R.layout.main);
		// 获取系统的ContentResolver对象
		contentResolver = getContentResolver();
	}

	public void query(View source)
	{
		// 调用ContentResolver的query()方法。
		// 实际返回的是该Uri对应的ContentProvider的query()的返回值
		Cursor c = contentResolver.query(uri, null
			, "query_where", null, null);
		Toast.makeText(this, "远程ContentProvide返回的Cursor为:" + c,
			Toast.LENGTH_LONG).show();
	}

	public void insert(View source)
	{
		ContentValues values = new ContentValues();
		values.put("name", "fkjava");
		// 调用ContentResolver的insert()方法。
		// 实际返回的是该Uri对应的ContentProvider的insert()的返回值
		Uri newUri = contentResolver.insert(uri, values);
		Toast.makeText(this, "远程ContentProvide新插入记录的Uri为:"
			+ newUri, Toast.LENGTH_LONG).show();
	}

	public void update(View source)
	{
		ContentValues values = new ContentValues();
		values.put("name", "fkjava");
		// 调用ContentResolver的update()方法。
		// 实际返回的是该Uri对应的ContentProvider的update()的返回值
		int count = contentResolver.update(uri, values
			, "update_where", null);
		Toast.makeText(this, "远程ContentProvide更新记录数为:"
			+ count, Toast.LENGTH_LONG).show();
	}

	public void delete(View source)
	{
		// 调用ContentResolver的delete()方法。
		// 实际返回的是该Uri对应的ContentProvider的delete()的返回值
		int count = contentResolver.delete(uri
			, "delete_where", null);
		Toast.makeText(this, "远程ContentProvide删除记录数为:"
			+ count, Toast.LENGTH_LONG).show();
	}


五,ContentProvider与ContentResolver的关系

ContentResolver对指定的Uri指定CRUD等数据操作,但是Uri并不是真正的数据中心,因此这些CRUD操作会委托给Uri对应的ContentProvider来实现。通常来说,A应用通过ContentResolver执行CRUD操作,这些操作都需要指定Uri参数,Android系统就根据该Uri找到对应的ContentProvider(该ContentProvider通常属于B应用),ContentProvider则负责实现CRUD方法,完成对底层数据的增删改查等操作,这样A应用就可以访问,修改B应用的数据了






评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值