Android组件之自定义ContentProvider
Android的数据存储有五种方式Shared Preferences、网络存储、文件存储、外储存储、SQLite,一般这些存储都只是在单独的一个应用程序之中达到一个数据的共享,有时候我们需要操作其他应用程序的一些数据,例如常见系统里的通讯录,短信,照片等等,所以云存储,通讯录,艳照门等等就孕育而生了。ContentProvider可以理解成内容提供者,也可以理解为一个接口,就是提供了一个供外部访问的接口,有的时候需要进行权限控制。
ContentProvider简介
ContentProvider向我们提供了我们在应用程序之前共享数据的一种机制,而我们知道每一个应用程序都是运行在不同的应用程序的,不同程序的之间数据共享是现实的需要,程序总不能使闭环的,Android中的ContentProvider外共享数据的好处是统一了数据的访问方式。简单总结说下:
- ContentProvider为存储和获取数据提供了统一的接口。ContentProvide对数据进行封装,不用关心数据存储的细节。使用表的形式来组织数据。
- 使用ContentProvider可以在不同的应用程序之间共享数据。
- Android为常见的一些数据提供了默认的ContentProvider(包括音频、视频、图片和通讯录等)。
说到了ContentProvider这么多好处,不能不说下Uri(Universal Resource Identifier)注意不是URL,通用资源标识符,看个简单的读取联系人的Uri,content://contacts/people,
- content://是前缀,固定的;
- contacts 主机名(或叫Authority)用于唯一标识这个ContentProvider,外部调用者可以根据这个标识来调用;
- people 路径(path)表示我们要操作的数据,路径的构建根据业务而定;
自定义ContentProvider
俗话说,欲善其事必先利其器,想要成为一个内容提供者,就先需要有数据,先建立一个SqlDbConncetion:
public class SqlDBConnection extends SQLiteOpenHelper {
	private static final String DbName ="Book.db";
	private static int version=1;
	public SqlDBConnection(Context context) {
		super(context, DbName, null, version);
	}
	@Override
	public void onCreate(SQLiteDatabase db) {
		// TODO Auto-generated method stub
		 String sqlString="create table Book (id integer primary key autoincrement,Name nvarchar(200),Title nvarchar(200))";
		 db.execSQL(sqlString);
	}
	@Override
	public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
		// TODO Auto-generated method stub
	}
}
上篇文章讲的junit测试这个时候可以拿过来使用一下初始化下数据:
public class BookCase extends AndroidTestCase {
	public void Intial() {
		SqlDBConnection dbConnection = new SqlDBConnection(getContext());
		SQLiteDatabase sqlDataBase = dbConnection.getWritableDatabase();
		long row = 0;
		for (int i = 0; i < 5; i++) {
			ContentValues values = new ContentValues();
			values.put("Name", "书籍"+i);
			values.put("Title", "标题" + i);
			row = sqlDataBase.insert("Book", null, values);
			Log.i("BookCase", "插入成功:" + row);
		}
	}
}
前面是基础工作,这个时候就可以建立一个自己的ContentProvider:
public class MyContentProvider extends ContentProvider {
	private static final String AUTHORITY = "com.example.googlecontentprovider.MyContentProvider";
	private static final int CONTENT_INSERT = 0;
	private static final int CONTENT_QUERY = 1;
	private static final int CONTENT_DELETE = 2;
	private static final int CONTENT_UPDATE = 3;
	private static final int CONTENT_QUERY_SINGLE = 4;
	private static UriMatcher uriMatcher;
	private SqlDBConnection dbConnection = null;
	static {
		uriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
		uriMatcher.addURI(AUTHORITY, "book/insert", CONTENT_INSERT);
		uriMatcher.addURI(AUTHORITY, "book/query", CONTENT_QUERY);
		uriMatcher.addURI(AUTHORITY, "book/delete", CONTENT_DELETE);
		uriMatcher.addURI(AUTHORITY, "book/update", CONTENT_UPDATE);
		uriMatcher.addURI(AUTHORITY, "book/query/#", CONTENT_QUERY_SINGLE);
	}
	@Override
	public boolean onCreate() {
		dbConnection = new SqlDBConnection(getContext());
		return true;
	}
	@Override
	public Cursor query(Uri uri, String[] projection, String selection,
			String[] selectionArgs, String sortOrder) {
		SQLiteDatabase dbDatabase = dbConnection.getWritableDatabase();
		switch (uriMatcher.match(uri)) {
		case CONTENT_QUERY:
			if (dbDatabase.isOpen()) {
				Cursor cursor=dbDatabase.query("Book", projection, selection, selectionArgs, null, null, null);
				return cursor;
			}
			break;
		case CONTENT_QUERY_SINGLE:
			if (dbDatabase.isOpen()) {
				long id=ContentUris.parseId(uri);
				Cursor cursor=dbDatabase.query("Book", projection, "id=?", new String[]{id+""}, null, null, null);
				return cursor;
			}
			break;
		default:
			break;
		}
		return null;
	}
	@Override
	public String getType(Uri uri) {
		// TODO Auto-generated method stub
		return null;
	}
	@Override
	public Uri insert(Uri uri, ContentValues values) {
		SQLiteDatabase dbDatabase = dbConnection.getWritableDatabase();
		switch (uriMatcher.match(uri)) {
		case CONTENT_INSERT:
			if (dbDatabase.isOpen()) {
				long id = dbDatabase.insert("Book", null, values);
				dbDatabase.close();
				return ContentUris.withAppendedId(uri, id);
			}
			break;
		default:
			break;
		}
		return null;
	}
	@Override
	public int delete(Uri uri, String selection, String[] selectionArgs) {
		// TODO Auto-generated method stub
		SQLiteDatabase dbDatabase = dbConnection.getWritableDatabase();
		switch (uriMatcher.match(uri)) {
		case CONTENT_DELETE:
			if (dbDatabase.isOpen()) {
				int count= dbDatabase.delete("Book", selection, selectionArgs);
				dbDatabase.close();
				return count;
			}
			break;
		default:
			break;
		}
		return 0;
	}
	@Override
	public int update(Uri uri, ContentValues values, String selection,
			String[] selectionArgs) {
		SQLiteDatabase dbDatabase = dbConnection.getWritableDatabase();
		switch (uriMatcher.match(uri)) {
		case CONTENT_UPDATE:
			if (dbDatabase.isOpen()) {
				int count= dbDatabase.update("Book", values, selection, selectionArgs);
				dbDatabase.close();
				return count;
			}
			break;
		default:
			break;
		}
		return 0;
	}
}
主机名是需要自己去AndroidManifest.xml文件中自己配置的,要求是唯一的,最好是用包名就好:
<provider android:name="com.example.googlecontentprovider.MyContentProvider"
android:authorities="com.example.googlecontentprovider.MyContentProvider"></provider>
如果觉得上面的那一串代码不是很好理解,下面调用的时候我会分别解释。
ContentResolver的使用
方法写在一个应用程序中调用属于正常,在另外一个程序中调用该程序的方法就是类似于接口了,下面先看原来初始化的数据:

重新新建一个Android测试项目,定义为BookCase,首先插入数据,定义一个Uri,这里面主机名就是上面定义的包名,book/insert与CONTENT_INSERT是对应的:
	public void bookInsert() {
		Uri uri = Uri
				.parse("content://com.example.googlecontentprovider.MyContentProvider/book/insert");
		ContentResolver resolver = getContext().getContentResolver();
		ContentValues values = new ContentValues();
		values.put("Name", "书籍5");
		values.put("Title", "标题5");
		uri = resolver.insert(uri, values);
		Log.i("BookCase", "Uri" + uri);
		long id = ContentUris.parseId(uri);
		Log.i("BookCase", "测试成功" + id);
	}
显示结果如下:

然后更新刚才插入的数据,同样的更具Code给Uri赋值,然后初始化一个ContentResolver,调用update方法:
	public void bookUpdate() {
		Uri uri = Uri
				.parse("content://com.example.googlecontentprovider.MyContentProvider/book/update");
		ContentResolver resolver = getContext().getContentResolver();
		ContentValues values=new ContentValues();
		values.put("Name", "修改");
		int count = resolver.update(uri, values, " id=?",new String[]{"10"});
		Log.i("BookCase", "更新了" + count + "行");
	}
结果如下:

删除插入的数据:
public void bookDelete() {
		Uri uri = Uri
				.parse("content://com.example.googlecontentprovider.MyContentProvider/book/delete");
		ContentResolver resolver = getContext().getContentResolver();
		String where =" id=?";
		String[] argString = {"10"};
		int count = resolver.delete(uri, where, argString);
		Log.i("BookCase", "删除了" + count + "行");
	}
结果如下:

查询所有的数据:
	public void bookQuery() {
		Uri uri = Uri
				.parse("content://com.example.googlecontentprovider.MyContentProvider/book/query");
		ContentResolver resolver = getContext().getContentResolver();
		Cursor  cursor=resolver.query(uri, new String[]{"id","Name","Title"}, null, null, null);
		if (cursor.getCount()>0) {
			while (cursor.moveToNext()) {
				int id=cursor.getInt(cursor.getColumnIndex("Id"));
				String nameString=cursor.getString(cursor.getColumnIndex("Name"));
				String titleString=cursor.getString(cursor.getColumnIndex("Title"));
				Log.i("BookCase", id+"---"+nameString+"---"+titleString);
			}
		}
	}
Log下的记录:

查询单条记录:
	public void bookQuerySingle() {
		Uri uri = Uri
				.parse("content://com.example.googlecontentprovider.MyContentProvider/book/query");
		ContentResolver resolver = getContext().getContentResolver();
		uri=ContentUris.withAppendedId(uri,1);
		Cursor  cursor=resolver.query(uri, new String[]{"id","Name","Title"}, null, null, null);
		if (cursor.getCount()>0) {
			while (cursor.moveToNext()) {
				int id=cursor.getInt(cursor.getColumnIndex("Id"));
				String nameString=cursor.getString(cursor.getColumnIndex("Name"));
				String titleString=cursor.getString(cursor.getColumnIndex("Title"));
				Log.i("BookCase", id+"---"+nameString+"---"+titleString);
			}
		}
	}
结果如图:

至此一个自定义ContentProvider全部搞定,如有不当,请各位多多指教~
Android组件之自定义ContentProvider的更多相关文章
- Android组件体系之ContentProvider使用注意事项
		1.数据访问机制 客户端/调用者通过getContentResolver调用,由ActivityThread.AMS获取到ContentProvider的代理,再通过这个代理对象调用服务端的实现(也即 ... 
- android开发之自定义组件
		android开发之自定义组件 一:自定义组件: 我认为,自定义组件就是android给我们提供的的一个空白的可以编辑的图片,它帮助我们实现的我们想要的界面,也就是通过自定义组件我们可以把我们要登入的 ... 
- 【Android 应用开发】 自定义组件 宽高适配方法, 手势监听器操作组件,  回调接口维护策略, 绘制方法分析 -- 基于 WheelView 组件分析自定义组件
		博客地址 : http://blog.csdn.net/shulianghan/article/details/41520569 代码下载 : -- GitHub : https://github.c ... 
- Android 组件ContentProvider
		Android 组件ContentProvider Android的数据存储有五种方式Shared Preferences.网络存储.文件存储.外储存储.SQLite,一般这些存储都仅仅是在单独的一个 ... 
- Android 四大组件学习之ContentProvider二
		上节学习了什么是ContentProvider.以及ContentProvider的作用.以及什么是URL.本节就对上节学习的知识做一个实践,也就是定义自己的ContentProvider 好.实践是 ... 
- 自定义ContentProvider的一些细节探究
		1. 适用范围 对于什么情况下才会用到自定义的ContentProvider,官方文档的Dev Guide是这样描述的: 如果你想要提供以下的一种或几种特性的时候你才需要构造一个ContentPr ... 
- Android组件的通讯——Intent
		转载:Android组件的通讯-Intent 1.概述 一个应用程序的三个核心组件——activities.services.broadcast receivers,都是通过叫做intents的消息激 ... 
- 自定义ContentProvider
		ContentProvider作为安卓的四大组件之一,在看开发中用到的频率远不如其他三个,以至于我都把这个东西给忘了,最近由于工作原因,不得不重新拾起来总结一下,那么今天就来说说自定义ContentP ... 
- 【Android开发日记】之入门篇(十二)——Android组件间的数据传输
		组件我们有了,那么我们缺少一个组件之间传递信息的渠道.利用Intent做载体,这是一个王道的做法.还有呢,可以利用文件系统来做数据共享.也可以使用Application设置全局数据,利用组件来进行控制 ... 
随机推荐
- CSUOJ 1008 Horcrux
			Description A Horcrux is an object in which a Dark wizard or witch has hidden a fragment of his or h ... 
- Linux-看完这篇Linux基本的操作就会了(转)
			前言 只有光头才能变强 这个学期开了Linux的课程了,授课的老师也是比较负责任的一位.总的来说也算是比较系统地学习了一下Linux了~~~ 本文章主要是总结Linux的基础操作以及一些简单的概念~如 ... 
- leetcode 奇偶链表 python
			要求空间复杂度O(1) 那就只能用指针不断改链表的指针, 不能建立新的内存 时间复杂度O(1) 一遍遍历 不能嵌套循环 我的思想是: 1 如果链表元素数量小于等于2个,那就无法操作 2 能操作的情况下 ... 
- sublime3176注册码破解汉化及常用插件
			官方网站下载地址:https://www.sublimetext.com/3 破解软件下载地址:https://www.lanzous.com/i1a7zfi 破解软件下载地址备用:https://d ... 
- loj#2718. 「NOI2018」归程
			题目链接 loj#2718. 「NOI2018」归程 题解 按照高度做克鲁斯卡尔重构树 那么对于询问倍增找到当前点能到达的高度最小可行点,该点的子树就是能到达的联通快,维护子树中到1节点的最短距离 s ... 
- [BZOJ5291][BJOI2018]链上二次求和(线段树)
			感觉自己做的麻烦了,但常数似乎不算差.(只是Luogu最慢的点不到2s本地要跑10+s) 感觉我的想法是最自然的,但不明白为什么网上似乎找不到这种做法.(不过当然所有的做法都是分类大讨论,而我的方法手 ... 
- Problem F: 铺地砖
			Description 元旦过去了,新年大酬宾活动也已经告一段落了.陈盖历望着堆在仓库的瓷砖,很无聊的他把这些瓷砖裁成很多1X1 1X2 1X3的小瓷砖,然后他把这些小瓷砖排在地上画的一个1*n的长方 ... 
- ZOJ 1940 Dungeon Master 三维BFS
			Dungeon Master Time Limit:1000MS Memory Limit:65536KB 64bit IO Format:%I64d & %I64u Desc ... 
- Xcode 小技巧
			1.手动添加 #warning ,在不确定的 bug.错误.待定代码处,手动添加 #warning 行,在编译时间提醒自己需要处理的地方. 2.由于 arrayWithObjects: 和 initW ... 
- [JAVA]  JAVA 类路径
			Java 类路径 类路径是所有包含类文件的路径的集合. 类路径中的目录和归档文件是搜寻类的起始点. 虚拟机搜寻类 搜寻jre/lib和jre/lib/ext目录中归档文件中所存放的系统类文件 搜寻再从 ... 
