Android组件之自定义ContentProvider

Android的数据存储有五种方式Shared Preferences、网络存储、文件存储、外储存储、SQLite,一般这些存储都只是在单独的一个应用程序之中达到一个数据的共享,有时候我们需要操作其他应用程序的一些数据,例如常见系统里的通讯录,短信,照片等等,所以云存储,通讯录,艳照门等等就孕育而生了。ContentProvider可以理解成内容提供者,也可以理解为一个接口,就是提供了一个供外部访问的接口,有的时候需要进行权限控制。

ContentProvider简介

ContentProvider向我们提供了我们在应用程序之前共享数据的一种机制,而我们知道每一个应用程序都是运行在不同的应用程序的,不同程序的之间数据共享是现实的需要,程序总不能使闭环的,Android中的ContentProvider外共享数据的好处是统一了数据的访问方式。简单总结说下:

  1. ContentProvider为存储和获取数据提供了统一的接口。ContentProvide对数据进行封装,不用关心数据存储的细节。使用表的形式来组织数据。
  2. 使用ContentProvider可以在不同的应用程序之间共享数据。
  3. Android为常见的一些数据提供了默认的ContentProvider(包括音频、视频、图片和通讯录等)。

说到了ContentProvider这么多好处,不能不说下Uri(Universal Resource Identifier)注意不是URL,通用资源标识符,看个简单的读取联系人的Uri,content://contacts/people,

  1. content://是前缀,固定的;
  2. contacts 主机名(或叫Authority)用于唯一标识这个ContentProvider,外部调用者可以根据这个标识来调用;
  3. 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全部搞定,如有不当,请各位多多指教~

时间: 11-03

Android组件之自定义ContentProvider的相关文章

自定义Android组件之组合方式创建密码框组件

Android中所有控件(也称组件)都继承自adnroid.view.View类,android.view.ViewGroup是View类的重要子类,绝大多书的布局类就继承自ViewGroup类. 附上一张基于Android Api21的View和Widget类图 自定义Android组件基本可以从2个入口着手,一是继承Viewe类拿起画笔和画布绘制组件,而是通过继承View的子类和组合已有的组件的方式构造自定义的组件. 本文通过自定义一个PassWordView组件来实现密码能够通过点击点选框

用自定义ContentProvider实现对数据库的增、删、改、查操作

本次对于数据库的操作是在不同的APP中进行的.下面的前四篇是在一个APP中,后面的是在另一个APP中 ------------------------------------------------------------------------------------------------- 首先自定义一个SqliteOpenHelper.完成对数据库的创建和数据库内table的创建 package com.example.content_provider_03; import androi

Android组件之间的信使——Intent

从一个Activity启动到另一个Activity可以使用startActivity()方法或者是startActivityForResult()方法 第一种:直接启动一个ActivityIntent intent = new Intent(Main.this, SecondActivity.class);startActivity(intent); 第二种:启动另一个Activity并返回结果作用:当从第二个Activity回跳到前一个Activity的时候,就不再需要使用startActiv

[Android]组件-进度条1

多式样ProgressBar 普通圆形ProgressBar 该类型进度条也就是一个表示运转的过程,例如发送短信,连接网络等等,表示一个过程正在执行中. 一般只要在XML布局中定义就可以了. <progressBar android:id="@+id/widget43" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layou

自定义组件-preference 自定义(设置-关于-系统更新)的preference

有些用户在安装好Android SDK后,打开Android SDK Manager下载API时一直显示"Done loading packages"却迟迟不能前进.自己也出现了这种情况,把自己成功解决此问题的方法分享给大家. 方法/步骤 用户需要首先打开已下载的安卓SDK管理器"Android SDK Manager",然后点击菜单栏中的"Tools"菜单选项,接下来只需选择"Options"选项即可打开设置窗体. 接下来,

【Android开发精要笔记】Android组件模型解析

Android组件模型解析 Android中的Mashup 将应用切分成不同类别的组件,通过统一的定位模型和接口标准将他们整合在一起,来共同完成某项任务.在Android的Mashup模式下,每个组件的功能都可以被充分的复用.来自不同应用的组件可以有机地结合在一起,共同完成任务. 基于Mashup的Android应用模型 三个基本要素:组件.连接.配置 接口就是实现单元.从代码来看,组件就是派生自特定接口或基类的子类的实现,如界面组件Activity就是指派生自android.app.Activ

Android UI之自定义——类似iOS的Tabbar

Android UI之自定义--类似iOS的Tabbar Tabbar最早出现在iOS,iOS中的TabBarController实现了这个功能,开发起来相当简单.现在的APP,大多数都会使用Tabbar来作为应用的功能导航,界面简单清晰.那么Android常见的实现是通过RadioGroup来实现,今天将带来自定义实现,补充RadioGroup实现的不足. 先看看常见的软件中的使用: 这个是高铁管家APP,大家应该非常熟悉.这个APP的首页底部就是一个类似iOS的Tabbar.这里就不多举例子

2015最流行的Android组件、工具、框架大全(转)

转自:2015最流行的Android组件.工具.框架大全 Android 是目前最流行的移动操作系统之一. 随着新版本的不断发布, Android的功能也日益强大, 涌现了很多流行的应用程序, 也催生了一大批的优秀的组件.本文试图将目前流行的组件收集起来以供参考, 如果你发现本文还没有列出的组件,欢迎在评论中贴出来,我会定期的更新本文. 部分图片需国内或许不能访问才能显示 很好的中文教程Google Android官方培训课程中文版 awesome-android, android列表. 另,g

【Android】19.3 ContentProvider及安卓进一步封装后的相关类

分类:C#.Android.VS2015: 创建日期:2016-03-08 一.简介 ContentProvider:内容提供程序. Android的ContentProvider与.NET框架的EF(Entity Framework)非常类似.在EF中,每个类表示数据库中的一个表,类中的每个属性对应表的字段,类的每个实例表示数据库表的一行记录.同样,在Android中,每个ContentProvider类的实例表示数据表的一行记录,ContentProvider实例集合中的每一项表示数据表中的