文档章节

使用android中ContentProvider组件对外共享以及监听ContentProvider中数据的变化

ht896632
 ht896632
发布于 2016/03/02 14:07
字数 1501
阅读 6
收藏 0
1.Contentprovider(组件)内容提供者:
    对外共享数据使用,可以把任意类型的数据进行对外共享;如:xml、数据库、txt等;好处是统一了数据对外的访问方式
    ContentProvider(内容提供者)的scheme已经由Android规定,scheme为:content://主机名(或Authority)用于唯一标识这个
    ContentProvider,外部调用这个可以根据这个标识来找到它。
    路径(Path)可以用来表示我们要操作的数据,路径的应用根据业务而定
    要操作user表中的id为1的记录,可以构建路径:/user/1
    要操作user表中的id为1的记录的age字段,可以构建路径:/user/1/age
    要操作user表中的所有记录,可以构建路径:/user

    如果要操作文件、xml或其它数据,可以这样命名路径:要操作xml文件中user节点下的age字段,可以构建路径:/user/age

使用方法,在使用ContentProvider时首先需要继承ContentProvider,因为ContentProvider是组件需要在应用清单文件中进行配置

示例:

import cn.kafei.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 UserProvider extends ContentProvider {
	private DBOpenHelper dbOpenHelper;//打开数据库
	private static final UriMatcher MATCHER = new UriMatcher(UriMatcher.NO_MATCH);//添加供匹配的URI
	private static final int USERS = 1;
	private static final int USER = 2;
	static {
		// 第一个参数:主机名,也就是在清单文件配置的provider的android:authorities
		// 第二参数:表示要匹配那个路径
		// 第三个参数:表示匹配码的标识,如果匹配成功就返回
		MATCHER.addURI("cn.kafei.provides.userprovider", "user", USERS);//匹配user表中的所有记录
		// #:表示数字,*号表示所有字符
		MATCHER.addURI("cn.kafei.provides.userprovider", "user/#", USER);//根据ID进行匹配
	}

	/**
	 * 当内容提供者的实例被创建出来之后被调用,只执行一次
	 */
	@Override
	public boolean onCreate() {
		dbOpenHelper = new DBOpenHelper(this.getContext());
		return true;
	}

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

	/**
	 * 可以供外部的应用往内容提供者插入数据 values:用来存放各个字段的值
	 */
	@Override
	public Uri insert(Uri uri, ContentValues values) {
		SQLiteDatabase db = dbOpenHelper.getWritableDatabase();
		switch (MATCHER.match(uri)) {
		case 1:
			// 如果主键是整型,那么这个行号就等于你的主键值
			long rowid = db.insert("user", "name", values);
			// content://cn.kafei.provides.userprovider/user/1
			// Uri
			// insertUri=Uri.parse("content://cn.kafei.provides.userprovider/user/"+rowid);
			Uri insertUri = ContentUris.withAppendedId(uri, rowid);
			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 1:
			num = db.delete("user", selection, selectionArgs);
			break;
		case 2:
			long rowid = ContentUris.parseId(uri);// 从URI中获取传入的ID
			String where = "id=" + rowid;
			if (selection != null && !"".equals(selection.trim())) {
				where += " and " + selection;
			}
			num = db.delete("user", where, selectionArgs);
			break;
		default:
			throw new IllegalArgumentException("this is Unknown Uri:" + uri);
		}
		return num;
	}

	/**
	 * 允许外部的应用对内容提供者中的数据进行更新
	 */
	@Override
	public int update(Uri uri, ContentValues values, String selection,
			String[] selectionArgs) {
		SQLiteDatabase db = dbOpenHelper.getWritableDatabase();
		int num = 0;
		switch (MATCHER.match(uri)) {
		case 1:
			num = db.update("user", values, selection, selectionArgs);
			break;
		case 2:
			long rowid = ContentUris.parseId(uri);// 从URI中提取传入的ID
			String where = "id=" + rowid;
			if (selection != null && !"".equals(selection.trim())) {
				where += " and " + selection;
			}
			num = db.update("user", values, 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();
		switch (MATCHER.match(uri)) {
		case 1:
			return db.query("user", projection, selection, selectionArgs, null,
					null, sortOrder);
		case 2:
			long rowid = ContentUris.parseId(uri);// 从URI中提取传入的ID
			String where = "id=" + rowid;
			if (selection != null && !"".equals(selection.trim())) {
				where += " and " + selection;
			}
			return db.query("user", projection, where, selectionArgs, null,
					null, sortOrder);
		default:
			throw new IllegalArgumentException("this is Unknown Uri:" + uri);
		}
	}
}
在应用清单文件中(application)标签内进行配置ContentProvider组件:

<!--android:name:表示ContentProvider的类名;android:authorities:表示主机名 ,主机名可以自定义,但是建议按照所提供的格式,以免引起重复-->
        <provider
            android:name=".UserProvider"
            android:authorities="cn.kafei.provides.userprovider" >
        </provider>
以上完成后就算ContentProvider配置完成
测试:

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 AccessContentProviderTest extends AndroidTestCase {
	private final String TAG="AccessContentProviderTest";
	/**
	 * 测试插入
	 */
	public void testInsert()throws Exception{
		Uri uri=Uri.parse("content://cn.kafei.provides.userprovider/user");//要操作的路径
		ContentResolver resolver=this.getContext().getContentResolver();
		ContentValues values=new ContentValues();
		values.put("name", "ht896632");
		values.put("age", "99");
		values.put("amount", "1234000000");
		resolver.insert(uri, values);
	}
	/**
	 * 测试删除
	 */
	public void testDelete()throws Exception{
		Uri uri=Uri.parse("content://cn.kafei.provides.userprovider/user/23");
		ContentResolver resolver=this.getContext().getContentResolver();
		resolver.delete(uri, null, null);
	}
	/**
	 * 测试更新
	 */
	public void testUpdate()throws Exception{
		Uri uri=Uri.parse("content://cn.kafei.provides.userprovider/user/22");
		ContentResolver resolver=this.getContext().getContentResolver();
		ContentValues values=new ContentValues();
		values.put("name", "李宏涛");
		
		resolver.update(uri, values, null, null);
	}
	/**
	 * 测试查询
	 */
	public void testQuery()throws Exception{
		Uri uri=Uri.parse("content://cn.kafei.provides.userprovider/user");
		ContentResolver resolver=this.getContext().getContentResolver();
		Cursor cursor=resolver.query(uri, null, null, null, "id desc");
		while(cursor.moveToNext()){
			String name=cursor.getString(cursor.getColumnIndex("name"));
			int id=cursor.getInt(cursor.getColumnIndex("id"));
			Log.i(TAG, "ID:"+id+" name:"+name);
		}
		cursor.close();
	}
}
注意:ContentProvider的注作用是对外共享数据。

2.监听ContentProvider数据中的变化

思路:首先往ContentProvider中添加数据,然后通过其它应用监听ContentProvider中的数据变化。

添加数据方法(使用一个应用往ContentProvider中添加数据):

public void add(View view){
    	Uri uri=Uri.parse("content://cn.kafei.provides.userprovider/user");//要操作的路径
		ContentResolver resolver=this.getContentResolver();
		ContentValues values=new ContentValues();
		values.put("name", "One");
		values.put("age", "101");
		values.put("amount", "9000");
		resolver.insert(uri, values);
    }
监听数据方法(放在一个应用的onCreate方法中,并且让这个应用保持在运行状态,否则监听不到):

Uri uri=Uri.parse("content://cn.kafei.provides.userprovider/user");//要操作的路径
        //第一个参数:要监听的URI
        //第二个参数:是否依赖
        //第三参数:表示监听对象
        this.getContentResolver().registerContentObserver(uri, true, new UserContentOberver(new Handler()));
       //以上代码放在onCreate方法中,监听时,作为监听的应该必须启动 
   /**
     * 监听数据方法
     */
    private class UserContentOberver extends ContentObserver{
		public UserContentOberver(Handler handler) {
			super(handler);
		}
		//只能粗略的监听到数据是否发生变化
    	@Override
    	public void onChange(boolean selfChange) {
    		//select * from user order by id desc limit 1
    		Uri uri=Uri.parse("content://cn.kafei.provides.userprovider/user");//要操作的路径
    		Cursor cursor=getContentResolver().query(uri, null, null, null, "id desc");
    		if(cursor.moveToFirst()){
    			String name=cursor.getString(cursor.getColumnIndex("name"));
    			Log.i(TAG, name);
    		}
    		
    	}
    }
注:ContentProvider、添加数据方法、监听数据方法在三个不同的应用中,连接他们之前的就是一个URI:
content://cn.kafei.provides.userprovider/user

© 著作权归作者所有

共有 人打赏支持
ht896632
粉丝 37
博文 53
码字总数 31361
作品 0
闵行
高级程序员
私信 提问
ContentProvider和Uri详解

问题引出: "/getScrollData" 的使用, public static final Uri INBOXCONTENTURI_SCROLLDATA = Uri .parse("content://" + SMSCONFIGAUTHORITY + "/" + SMSINBOXTABLE_NAME + "/getScrollDa......

eric_zhang
2012/03/23
0
0
[Android] ContentProvider和Uri详解

一、使用ContentProvider(内容提供者)共享数据 ContentProvider在android中的作用是对外共享数据,也就是说你可以通过ContentProvider把应用中的数据共享给其他应用访问,其他应用可以通过...

枫兮兮
2014/03/03
0
0
Android使用内容提供者方式进行存储

内容提供者(ContentProvider)主要作用是对外共享数据,如果数据通过内容提供者对外共享了,那么其他应用就可以从内容提供者中查询到数据,并且可更新数据、删除数据、添加数据,如果采用文件...

IBMdW
2011/09/26
715
0
android数据存储与访问之使用ContentProvider

ContentProvider简介及其好处 简介: ContentProvider 在android中的作用是对外共享数据,也就是说你可以通过ContentProvider把应用中的数据共享给其他应用访问,其他应用可以通过ContentProv...

长平狐
2012/09/03
219
0
android四大组件--ContentProvider详解

一、相关ContentProvider概念解析: 1、ContentProvider简介 在Android官方指出的Android的数据存储方式总共有五种,分别是:Shared Preferences、网络存储、文件存储、外储存储、SQLite。但...

大凉龙雀
2015/10/31
0
0

没有更多内容

加载失败,请刷新页面

加载更多

Linux syslog相关函数详解

介绍 syslog是Unix系统的日志系统。可以将日志记录在本地系统中。 一个完整的syslong日志包含如下信息:程序模块 | 严重性 | 时间 | 主机名 | 进程名 | 进程ID | 正文。 syslong相关函数 1....

RongJinhui0
17分钟前
0
0
使用nsenter工具进入Docker容器

查看本机装没有nsenter whereis nsenter或者whatis nsenter 未安装先安装,网上有很多这样的脚本 vi nsenter.sh#!/bin/bashcurl https://www.kernel.org/pub/linux/utils/util-linux/v2....

问题终结者
18分钟前
1
0
MaxCompute安全管理指南-基础篇

背景及目的 方便和辅助MaxCompute的project owner或安全管理员进行project的日常安全运维,保障数据安全。 MaxCompute有安全模型,DataWorks也有安全模型,当通过DataWorks使用MaxCompute,而...

阿里云云栖社区
18分钟前
0
0
Retrofit设计模式源码解析

因为Retrofit做到了很强的解耦,因此就一定需要用到很多设计模式。所以,我觉得,通过阅读Retrofit源码来学习设计模式是再好不过的设计模式学习方法了。 大致看了一圈Retrofit源码,受益匪浅...

亭子happy
18分钟前
3
0
哈夫曼编码

哈夫曼编码的基本思想是以字符的使用频率作为权构建一颗哈夫曼树,然后利用 哈夫曼树对字符进行编码 哈夫曼算法采用的贪心策略是每次从树的集合中取出没有双亲权值最小的两棵作为左右子树, ...

writeademo
20分钟前
1
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部