当应用继承ContentProvider类,并重写该类用于提供数据和存储数据的方法,就可以向其他应用共享其数据。以前我们学习过文件的操作模式,通过指定文件的操作模式为Context.MODE_WORLD_READABLE 或Context.MODE_WORLD_WRITEABLE同样可以对外共享数据,但数据的访问方式会因数据存储的方式而不同,如:采用xml文件对外共享数据,需要进行xml解析来读写数据;采用sharedpreferences共享数据,需要使用sharedpreferences API读写数据。而使用ContentProvider共享数据的好处是统一了数据访问方式。

当应用需要通过ContentProvider对外共享数据时,第一步需要继承ContentProvider并重写下面方法:

public class PersonContentProvider extends ContentProvider{
   public boolean onCreate()
   public Uri insert(Uri uri, ContentValues values)
   public int delete(Uri uri, String selection, String[] selectionArgs)
   public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs)
   public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder)
   public String getType(Uri uri)}



第二步需要在AndroidManifest.xml使用<provider>对该ContentProvider进行配置,为了能让其他应用找到该ContentProvider, ContentProvider采用

了authorities(主机名/域名)对它进行唯一标识,你可以把ContentProvider看作是一个网站(想想,网站也是提供数据者),authorities就是他的域名:

<manifest .... >
    <application android:icon="@drawable/icon" android:label="@string/app_name">
        <provider android:name=".PersonContentProvider" android:authorities="cn.itcast.providers.personprovider"/>
    </application>
</manifest>



注意:一旦应用继承了ContentProvider类,后面我们就会把这个应用称为ContentProvider(内容提供者)。

Uri代表了要操作的数据,Uri主要包含了两部分信息:1》需要操作的ContentProvider ,2》对ContentProvider中的什么数据进行操作,一个Uri由以下几部分组成:


android 共享元素item改变 android 共享数据_数据


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")

因为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类用于获取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

ContentProvider类主要方法的作用:

public boolean onCreate()

该方法在ContentProvider创建后就会被调用, Android开机后, ContentProvider在其它应用第一次访问它时才会被创建。

public Uriinsert(Uri uri, ContentValues values)

该方法用于供外部应用往ContentProvider添加数据。

public int delete(Uri uri, String selection,String[] selectionArgs)

该方法用于供外部应用从ContentProvider删除数据。

public int update(Uri uri, ContentValues values, Stringselection, String[] selectionArgs)

该方法用于供外部应用更新ContentProvider中的数据。

public Cursorquery(Uri uri, String[]projection, String selection, String[] selectionArgs, String sortOrder)

该方法用于供外部应用从ContentProvider中获取数据。

public String getType(Uri uri)

该方法用于返回当前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”。


使用ContentResolver操作ContentProvider中的数据


当外部应用需要对ContentProvider中的数据进行添加、删除、修改和查询操作时,可以使用ContentResolver 类来完成,要获取ContentResolver 对象,可以使用Activity提供的getContentResolver()方法。 ContentResolver 类提供了与ContentProvider类相同签名的四个方法:

public Uriinsert(Uri uri, ContentValues values)

该方法用于往ContentProvider添加数据。

public int delete(Uri uri, String selection,String[] selectionArgs)

该方法用于从ContentProvider删除数据。

public int update(Uri uri, ContentValues values, Stringselection, String[] selectionArgs)

该方法用于更新ContentProvider中的数据。

public Cursorquery(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的记录。

1、工程结构:

android 共享元素item改变 android 共享数据_android_02

2、内容监听者的清单配置

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="cn.huangjie.db"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk android:minSdkVersion="8" />

    <application
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name" >
        <activity
            android:label="@string/app_name"
            android:name=".DbActivity" >
            <intent-filter >
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <!-- 加上这行相当于引入包的意思 -->
        <uses-library android:name="android.test.runner" />
        
        <!-- android:authorities内容提供者的唯一标识,取名最好遵守一定的规则 -->
        <provider android:name=".PersonProvider" android:authorities="cn.huangjie.provides.personprovider">
            
        </provider>
    </application>
	<instrumentation android:name="android.test.InstrumentationTestRunner"
  	android:targetPackage="cn.huangjie.db" android:label="Tests for My App" />
    <!-- 上面targetPackage指定的包要和应用的package相同  -->    
</manifest>



3、内容提供者的实现

package cn.huangjie.db;

import cn.huangjie.service.DBOpenHelper;
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 DBOpenHelper dbOpenHelper;
	/**
	 * UriMatcher需要传入一个数字,传入的数字代表如果跟输入进来的uri不匹配
	 * 的话那么返回的匹配码,这个输入参数就是不匹配的时候返回出来的不匹配码
	 */
	private static final UriMatcher MATCHER = new UriMatcher(UriMatcher.NO_MATCH);
	//匹配码
	private static final int PERSONS = 1;
	private static final int PERSON = 2;
	static{
		MATCHER.addURI("cn.huangjie.provides.personprovider", "person", PERSONS);
		//在android中,#代表数字,*代表任意字符
		MATCHER.addURI("cn.huangjie.provides.personprovider", "person/#", PERSON);
	}

	/**
	 * 该方法由系统调用的,当这个内容提供者的实例被创建出来之后这个方法被调用
	 */
	@Override
	public boolean onCreate() {
		dbOpenHelper = new DBOpenHelper(this.getContext());
		return true;
	}

	/**
	 * 返回你目前要操作数据的内容类型
	 * 例如:txt,html,mp3
	 */
	@Override
	public String getType(Uri uri) {
		switch(MATCHER.match(uri)){
			case PERSONS:
				return "vnd.android.cursor.dir/";
			case PERSON:
				return "vnd.android.cursor.item/";
			default:
				throw new IllegalArgumentException("this is Unknown Uri:" + uri);
		}
	}

	/**
	 * 允许外部的应用对内容提供者进行插入
	 */
	@Override
	public Uri insert(Uri uri, ContentValues values) {
		SQLiteDatabase db = dbOpenHelper.getWritableDatabase();
		switch(MATCHER.match(uri)){
			case PERSONS:
				//返回行号,数据库的RecNo字段不是行号,当行号跟
				//如果主键是整型并且是自增长的话那么这个行号是等于主键值
				long rowid = db.insert("person", "name", values);
				// content://cn.huangjie.provides.personprovider/person/10
				//以下两种方式进行拼接的uri都可以
	//			Uri insertUri = Uri.parse("content://cn.huangjie.provides.personprovider/person/" + rowid);
				Uri insertUri = ContentUris.withAppendedId(uri, rowid);
				//发出数据变化通知,第二个参数为变化监听者,这里没有监听者
				//这样检测此数据的应用会得到相应的通知
				this.getContext().getContentResolver().notifyChange(uri, null);
				return insertUri;
			default:
				throw new IllegalArgumentException("this is Unknown Uri:" + uri);
		}
	}
	
	/**
	 * 允许外部的应用对内容提供者进行删除
	 * 返回这个操作所影响的记录数
	 */
	@Override
	public int delete(Uri uri, String selection, String[] selectionArgs) {
		SQLiteDatabase db = dbOpenHelper.getWritableDatabase();
		int num = 0;
		switch(MATCHER.match(uri)){
			case PERSONS:
				num = db.delete("person", selection, selectionArgs);
				break;
			case PERSON:
				//解析出/person/10这种形式的id值
				long rowid = ContentUris.parseId(uri);
				String where = " personid = " + rowid;
				if(selection != null && selection.length() > 0){
					where += selection;
				}
				num = db.delete("person", where, selectionArgs);
				break;
			default:
				throw new IllegalArgumentException("this is Unknown Uri:" + uri);
		}
		return num;
	}

	/**
	 * 可以供外部的应用查询内容提供者的数据
	 */
	@Override
	public Cursor query(Uri uri, String[] projection, String selection,
			String[] selectionArgs, String sortOrder) {
		SQLiteDatabase db = dbOpenHelper.getReadableDatabase();
		Cursor cursor = null;
		switch(MATCHER.match(uri)){
			case PERSONS:
				cursor = db.query("person", projection, selection, selectionArgs, null, null, sortOrder);
				break;
			case PERSON:
				//解析出/person/10这种形式的id值
				long rowid = ContentUris.parseId(uri);
				String where = " personid = " + rowid;
				if(selection != null && selection.length() > 0){
					where += selection;
				}
				cursor = db.query("person", projection, where, selectionArgs, null, null, sortOrder);
				break;
			default:
				throw new IllegalArgumentException("this is Unknown Uri:" + uri);
		}
		return cursor;
	}

	/**
	 * 允许外部的应用对内容提供者进行更新
	 */
	@Override
	public int update(Uri uri, ContentValues values, String selection,
			String[] selectionArgs) {
		SQLiteDatabase db = dbOpenHelper.getWritableDatabase();
		int num = 0;
		switch(MATCHER.match(uri)){
			case PERSONS:
				num = db.update("person", values, selection, selectionArgs);
				break;
			case PERSON:
				//解析出/person/10这种形式的id值
				long rowid = ContentUris.parseId(uri);
				String where = " personid = " + rowid;
				if(selection != null && selection.length() > 0){
					where += selection;
				}
				num = db.update("person", values,  where, selectionArgs);
				break;
			default:
				throw new IllegalArgumentException("this is Unknown Uri:" + uri);
		}
		return num;
	}

}



4、测试书写的内容提供者

package cn.huangjie.test;

import android.content.ContentResolver;
import android.content.ContentValues;
import android.database.Cursor;
import android.net.Uri;
import android.test.AndroidTestCase;
import android.util.Log;

public class ContentProvideTest extends AndroidTestCase{
	
	
	public void testInsert(){
		Uri uri = Uri.parse("content://cn.huangjie.provides.personprovider/person");
		//访问内容提供者
		ContentResolver resolver = this.getContext().getContentResolver();
		ContentValues values = new ContentValues();
		values.put("name", "jiehuang");
		values.put("age", 545);
		resolver.insert(uri, values);
	}
	
	public void testDelete(){
		Uri uri = Uri.parse("content://cn.huangjie.provides.personprovider/person/30");
		//访问内容提供者
		ContentResolver resolver = this.getContext().getContentResolver();
		int num = resolver.delete(uri, null, null);
		Log.i("ContentProvide", "删除影响条数:"+num);
	}

	public void testUpdate(){
		Uri uri = Uri.parse("content://cn.huangjie.provides.personprovider/person/31");
		//访问内容提供者
		ContentResolver resolver = this.getContext().getContentResolver();
		ContentValues values = new ContentValues();
		values.put("name", "jieh");
		values.put("age", 54);
		int num = resolver.update(uri, values, null, null);
		Log.i("ContentProvide", "更新影响条数:"+num);
	}

	public void testQuery(){
		Uri uri = Uri.parse("content://cn.huangjie.provides.personprovider/person");
		//访问内容提供者
		ContentResolver resolver = this.getContext().getContentResolver();
		
		Cursor cursor = resolver.query(uri, null, null, null, "personid desc");
		while(cursor.moveToNext()){
			int id = cursor.getInt(cursor.getColumnIndex("personid"));
			String name = cursor.getString(cursor.getColumnIndex("name"));
			Log.i("ContentProvide", id+"#"+name);
		}
	}
}



1、此外我们还可以监听内容提供者数据的变化,该应用场景如下:

android 共享元素item改变 android 共享数据_数据_03

2、首先我们需要在内容提供者实现类中增加一个监听的事件

android 共享元素item改变 android 共享数据_ide_04

3、在A应用中增加一个按钮,点击按钮往内容提供者中增加一条数据

界面配置文件:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical" >

    <Button
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@string/button"
        android:onClick="insert" /><!-- 调用Activity类里面的insert方法 -->

</LinearLayout>

数值配置文件:


<?xml version="1.0" encoding="utf-8"?>
<resources>

    <string name="hello">Hello World, AappActivity!</string>
    <string name="app_name">A应用</string>
	<string name="button">往内容提供者添加数据</string>
</resources>



Activity类:

package cn.huangjie.aapp;

import android.app.Activity;
import android.content.ContentResolver;
import android.content.ContentValues;
import android.net.Uri;
import android.os.Bundle;
import android.view.View;

public class AappActivity extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        
    }
    public void insert(View v){
    	Uri uri = Uri.parse("content://cn.huangjie.provides.personprovider/person");
		//访问内容提供者
		ContentResolver resolver = this.getContentResolver();
		ContentValues values = new ContentValues();
		values.put("name", "A_App");
		values.put("age", 444);
		resolver.insert(uri, values);
    }
}



4、在B应用中注册一个监听器来对插入的内容提供者加入监听

package cn.huangjie.other;

import android.app.Activity;
import android.database.ContentObserver;
import android.database.Cursor;
import android.net.Uri;
import android.os.Bundle;
import android.os.Handler;
import android.util.Log;

public class OtherActivity extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        Uri uri = Uri.parse("content://cn.huangjie.provides.personprovider/person");
        //第二个参数true表示依赖
        this.getContentResolver().registerContentObserver(uri, true, 
        		new PersonContentObserver(new Handler()));
    }
    
    private class PersonContentObserver extends ContentObserver{

		public PersonContentObserver(Handler handler) {
			super(handler);
		}

		/**
		 * 这个方法只探测到改变了,但是不知道是修改还是增加还是删除等
		 * 如果是增加的话则从数据库中读取一条最新的记录便是增加的
		 */
		@Override
		public void onChange(boolean selfChange) {
			//sql语句:select * from person order by personid desc limit 1
			Uri uri = Uri.parse("content://cn.huangjie.provides.personprovider/person");
			Cursor cursor = getContentResolver().query(uri, null, null, null, "personid desc limit 1");
			if(cursor.moveToFirst()){//如果有一条记录
				String name = cursor.getString(cursor.getColumnIndex("name"));
				Log.i("OtherActivity", "检测到的数据:"+name);
			}
		}
		
    }
}



运行结果:

android 共享元素item改变 android 共享数据_ide_05