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全部搞定,如有不当,请各位多多指教~

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

  1. Android组件体系之ContentProvider使用注意事项

    1.数据访问机制 客户端/调用者通过getContentResolver调用,由ActivityThread.AMS获取到ContentProvider的代理,再通过这个代理对象调用服务端的实现(也即 ...

  2. android开发之自定义组件

    android开发之自定义组件 一:自定义组件: 我认为,自定义组件就是android给我们提供的的一个空白的可以编辑的图片,它帮助我们实现的我们想要的界面,也就是通过自定义组件我们可以把我们要登入的 ...

  3. 【Android 应用开发】 自定义组件 宽高适配方法, 手势监听器操作组件, 回调接口维护策略, 绘制方法分析 -- 基于 WheelView 组件分析自定义组件

    博客地址 : http://blog.csdn.net/shulianghan/article/details/41520569 代码下载 : -- GitHub : https://github.c ...

  4. Android 组件ContentProvider

    Android 组件ContentProvider Android的数据存储有五种方式Shared Preferences.网络存储.文件存储.外储存储.SQLite,一般这些存储都仅仅是在单独的一个 ...

  5. Android 四大组件学习之ContentProvider二

    上节学习了什么是ContentProvider.以及ContentProvider的作用.以及什么是URL.本节就对上节学习的知识做一个实践,也就是定义自己的ContentProvider 好.实践是 ...

  6. 自定义ContentProvider的一些细节探究

    1.   适用范围 对于什么情况下才会用到自定义的ContentProvider,官方文档的Dev Guide是这样描述的: 如果你想要提供以下的一种或几种特性的时候你才需要构造一个ContentPr ...

  7. Android组件的通讯——Intent

    转载:Android组件的通讯-Intent 1.概述 一个应用程序的三个核心组件——activities.services.broadcast receivers,都是通过叫做intents的消息激 ...

  8. 自定义ContentProvider

    ContentProvider作为安卓的四大组件之一,在看开发中用到的频率远不如其他三个,以至于我都把这个东西给忘了,最近由于工作原因,不得不重新拾起来总结一下,那么今天就来说说自定义ContentP ...

  9. 【Android开发日记】之入门篇(十二)——Android组件间的数据传输

    组件我们有了,那么我们缺少一个组件之间传递信息的渠道.利用Intent做载体,这是一个王道的做法.还有呢,可以利用文件系统来做数据共享.也可以使用Application设置全局数据,利用组件来进行控制 ...

随机推荐

  1. CSUOJ 1901 赏赐 OR 灾难 单调栈

    Description 大G南征北战终于打下了大片土地成立了G国,大G在开国大典上传召帮助自己南征北战的三大开国元勋小A,小B,小C进殿,并要赏赐三人大量宝物以显示天恩浩荡.大G在征服其他国家的时候抢 ...

  2. Java 中的数据类型

    我们学习Java就是为了编写程序完成功能,而什么是程序呢?程序 = 数据结构 + 算法. 分开看,数据结构指的是数据与数据之间的关系,那我们先来了解一下Java中的数据都是怎么表示的呢 ?也就是说数据 ...

  3. window下完全删除nodejs

    1.从卸载程序卸载程序和功能. 2.重新启动(或者您可能会从任务管理器中杀死所有与节点相关的进程). 3.寻找这些文件夹并删除它们(及其内容)(如果还有).根据您安装的版本,UAC设置和CPU架构,这 ...

  4. 01-学前入门.Net两种交换模式

    C/S:客户机(Client)/服务器模式(Server)Winfrom应用程序 B/S:浏览器(Browser)/服务器模式(Server)Internet应用模式

  5. 【WIN10】VisualStateManager使用說明

    Demo下載:http://yunpan.cn/cFjgPtWRHKH9H  访问密码 c4b7 顧名思義,視圖狀態管理器. 在WPF中,它的功能似乎更強大.在UWP中,閹割了GotElementSt ...

  6. 「HNOI2018」毒瘤

    「HNOI2018」毒瘤 解题思路 先考虑只有一棵树的情况,经典独立集计数. \[ dp[u][0]=\prod (dp[v][0]+dp[v][1]) \\ dp[u][1]=\prod dp[v] ...

  7. HDU 3282 Running Median 动态中位数,可惜数据范围太小

    Running Median Time Limit: 1 Sec  Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/showproblem.php?pi ...

  8. zoj 3629 Treasure Hunt IV 打表找规律

    H - Treasure Hunt IV Time Limit:2000MS     Memory Limit:65536KB     64bit IO Format:%lld & %llu ...

  9. ASP.NET与MVC架构区别总结

    1)ASP.NET Webforms Behind Code利于快速开发,方便可视化操作. 2)ASP.NET 使用了“基于视图”的解决方案去应对“基于行为”的需求,它处理了客户端的请求,IIS将请求 ...

  10. Oracle sql语句中(+)作用

    select * from operator_info o, group_info g  where o.group_id = g.group_id(+); 理解:    + 表示补充,即哪个表有加号 ...