Android开发14——监听内容提供者ContentProvider的数据变化

简介:

 

一、提出需求

有A,B,C三个应用,B中的数据需要被共享,所以B中定义了内容提供者ContentProvider;A应用修改了B应用的数据,插入了一条数据。有这样一个需求,此时C应用需要得到数据被修改的通知并处理相应操作。

 

 

二、示例代码

A应用
/**
 * 对内容提供者进行操作
 * 
 * @author XY
 * 
 */
public class MainActivity extends Activity
{

	@Override
	public void onCreate(Bundle savedInstanceState)
	{
		super.onCreate(savedInstanceState);
		setContentView(R.layout.main);
	}

	public void insert(View v)
	{
		Uri uri = Uri.parse("content://cn.xyCompany.providers.personProvider/person");
		ContentResolver resolver = this.getContentResolver();
		ContentValues values = new ContentValues();
		values.put("name", "xy_new_new");
		values.put("phone", "xy_new_111");
		resolver.insert(uri, values);
	}
}


B应用
package cn.xy.cotentProvider.app.providers;
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;
import android.util.Log;
import cn.xy.cotentProvider.service.DBOpeningHelper;

/**
 * @author XY
 * 
 */
public class PersonProvider extends ContentProvider
{
	private DBOpeningHelper dbHelper;

	// 若不匹配采用UriMatcher.NO_MATCH(-1)返回
	private static final UriMatcher MATCHER = new UriMatcher(UriMatcher.NO_MATCH);

	// 匹配码
	private static final int CODE_NOPARAM = 1;
	private static final int CODE_PARAM = 2;

	static
	{
		// 对等待匹配的URI进行匹配操作,必须符合cn.xyCompany.providers.personProvider/person格式
		// 匹配返回CODE_NOPARAM,不匹配返回-1
		MATCHER.addURI("cn.xyCompany.providers.personProvider", "person", CODE_NOPARAM);

		// #表示数字 cn.xyCompany.providers.personProvider/person/10
		// 匹配返回CODE_PARAM,不匹配返回-1
		MATCHER.addURI("cn.xyCompany.providers.personProvider", "person/#", CODE_PARAM);
	}

	@Override
	public boolean onCreate()
	{
		dbHelper = new DBOpeningHelper(this.getContext());
		return true;
	}

	/**
	 * 外部应用向本应用插入数据
	 */
	@Override
	public Uri insert(Uri uri, ContentValues values)
	{
		SQLiteDatabase db = dbHelper.getWritableDatabase();
		switch (MATCHER.match(uri))
		{
			case CODE_NOPARAM:
				// 若主键值是自增长的id值则返回值为主键值,否则为行号,但行号并不是RecNo列
				long id = db.insert("person", "name", values); 
				Uri insertUri = ContentUris.withAppendedId(uri, id); 
				// 发出变化通知(非必须)设监听者为null。
				// 若设置某个监听者则不管有多少个监听者,该监听者一定可以获得该通知
				getContext().getContentResolver().notifyChange(uri, null); 
				return insertUri;
			default:
				throw new IllegalArgumentException("this is unkown uri:" + uri);
		}
	}
	......
}


C应用
package cn.xt.contentProvider.lisenter;
import android.app.Activity;
import android.content.ContentResolver;
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 MainActivity extends Activity
{
	@Override
	public void onCreate(Bundle savedInstanceState)
	{
		super.onCreate(savedInstanceState);
		setContentView(R.layout.main);
		Uri uri = Uri.parse("content://cn.xyCompany.providers.personProvider/person");
		ContentResolver resolver = this.getContentResolver();
		resolver.registerContentObserver(uri, true, new PersonContentObserver(new Handler()));
	}

	private class PersonContentObserver extends ContentObserver
	{
		public PersonContentObserver(Handler handler)
		{
			super(handler);
		}

		// 得到数据的变化通知,该方法只能粗略知道数据的改变,并不能判断是哪个业务操作进行的改变
		@Override
		public void onChange(boolean selfChange)
		{
			// select * from person order by id desc limit 1 // 取得最近插入的值(序号大——>小并取第一个)
			Uri uri = Uri.parse("content://cn.xyCompany.providers.personProvider/person");
			ContentResolver resolver = MainActivity.this.getContentResolver();
			Cursor cursor = resolver.query(uri, null, null, null, "id desc limit 1");
			if(cursor.moveToFirst())
			{
				String name = cursor.getString(cursor.getColumnIndex("name"));
				Log.i("lisenter", name);
			}
		}
	}
}

关于contentProvider的基本使用,请参看本博客博文《Android开发13——内容提供者ContentProvider的基本使用》

 

目录
相关文章
|
21天前
|
Java Android开发
Android 开发获取通知栏权限时会出现两个应用图标
Android 开发获取通知栏权限时会出现两个应用图标
12 0
|
3天前
|
Android开发 开发者
Android网络和数据交互: 请解释Android中的AsyncTask的作用。
Android's AsyncTask simplifies asynchronous tasks for brief background work, bridging UI and worker threads. It involves execute() for starting tasks, doInBackground() for background execution, publishProgress() for progress updates, and onPostExecute() for returning results to the main thread.
3 0
|
3天前
|
网络协议 安全 API
Android网络和数据交互: 什么是HTTP和HTTPS?在Android中如何进行网络请求?
HTTP和HTTPS是网络数据传输协议,HTTP基于TCP/IP,简单快速,HTTPS则是加密的HTTP,确保数据安全。在Android中,过去常用HttpURLConnection和HttpClient,但HttpClient自Android 6.0起被移除。现在推荐使用支持TLS、流式上传下载、超时配置等特性的HttpsURLConnection进行网络请求。
5 0
|
12天前
|
XML 开发工具 Android开发
构建高效的安卓应用:使用Jetpack Compose优化UI开发
【4月更文挑战第7天】 随着Android开发不断进化,开发者面临着提高应用性能与简化UI构建流程的双重挑战。本文将探讨如何使用Jetpack Compose这一现代UI工具包来优化安卓应用的开发流程,并提升用户界面的流畅性与一致性。通过介绍Jetpack Compose的核心概念、与传统方法的区别以及实际集成步骤,我们旨在提供一种高效且可靠的解决方案,以帮助开发者构建响应迅速且用户体验优良的安卓应用。
|
14天前
|
Java Android开发
Android开发之使用OpenGL实现翻书动画
本文讲述了如何使用OpenGL实现更平滑、逼真的电子书翻页动画,以解决传统贝塞尔曲线方法存在的卡顿和阴影问题。作者分享了一个改造后的外国代码示例,提供了从前往后和从后往前的翻页效果动图。文章附带了`GlTurnActivity`的Java代码片段,展示如何加载和显示书籍图片。完整工程代码可在作者的GitHub找到:https://github.com/aqi00/note/tree/master/ExmOpenGL。
19 1
Android开发之使用OpenGL实现翻书动画
|
14天前
|
Android开发 开发者
Android开发之OpenGL的画笔工具GL10
这篇文章简述了OpenGL通过GL10进行三维图形绘制,强调颜色取值范围为0.0到1.0,背景和画笔颜色设置方法;介绍了三维坐标系及与之相关的旋转、平移和缩放操作;最后探讨了坐标矩阵变换,包括设置绘图区域、调整镜头参数和改变观测方位。示例代码展示了如何使用这些方法创建简单的三维立方体。
12 1
Android开发之OpenGL的画笔工具GL10
|
17天前
|
XML Java Android开发
Android每点击一次按钮就添加一条数据
Android每点击一次按钮就添加一条数据
21 1
|
21天前
|
Android开发
Android开发小技巧:怎样在 textview 前面加上一个小图标。
Android开发小技巧:怎样在 textview 前面加上一个小图标。
10 0
|
21天前
|
Android开发
Android 开发 pickerview 自定义选择器
Android 开发 pickerview 自定义选择器
12 0
|
27天前
|
Java Android开发
Android开发系列全套课程
本系列课程面向有java基础,想进入企业从事android开发的计算机专业者。学习搭配实战案例,高效掌握岗位知识。
18 1