ContentProvider简要
1.什么是ContentProvider
数据库在Android其中是私有的,当然这些数据包含文件数据和数据库数据以及一些其它类型的数据。
不能将数据库设为WORLD_READABLE,每一个数据库都仅仅能创建它的包訪问,
这意味着仅仅有由创建数据库的进程可訪问它。假设须要在进程间传递数据,
则能够使用AIDL/Binder或创建一个ContentProvider,可是不能跨越进程/包边界直接来使用数据库。
一个Content Provider类实现了一组标准的方法接口,从而可以让其它的应用保存或读取此Content Provider的各种数据类型。
也就是说,一个程序能够通过实现一个Content Provider的抽象接口将自己的数据暴露出去。
外界根本看不到,也不用看到这个应用暴露的数据在应用其中是怎样存储的,或者是用数据库存储还是用文件存储,还是通过网上获得,这些一切都不重要,
重要的是外界能够通过这一套标准及统一的接口和程序里的数据打交道,能够读取程序的数据,也能够删除程序的数据,
当然,中间也会涉及一些权限的问题。下边列举一些较常见的接口,这些接口例如以下所看到的。
· query(Uri uri, String[] projection, String selection, String[] selectionArgs,String sortOrder):通过Uri进行查询,返回一个Cursor。
· insert(Uri url, ContentValues values):将一组数据插入到Uri 指定的地方。
· update(Uri uri, ContentValues values, String where, String[] selectionArgs):更新Uri指定位置的数据。
· delete(Uri url, String where, String[] selectionArgs):删除指定Uri而且符合一定条件的数据。
2.什么是ContentResolver
外界的程序通过ContentResolver接口能够訪问ContentProvider提供的数据,在Activity其中通过getContentResolver()能够得到当前应用的 ContentResolver实例。
ContentResolver提供的接口和ContentProvider中须要实现的接口相应,主要有下面几个。
· query(Uri uri, String[] projection, String selection, String[] selectionArgs,String sortOrder):通过Uri进行查询,返回一个Cursor。
· insert(Uri url, ContentValues values):将一组数据插入到Uri 指定的地方。
· update(Uri uri, ContentValues values, String where, String[] selectionArgs):更新Uri指定位置的数据。
· delete(Uri url, String where, String[] selectionArgs):删除指定Uri而且符合一定条件的数据。
3.ContentProvider和ContentResolver中用到的Uri
在ContentProvider和 ContentResolver其中用到了Uri的形式通常有两种,一种是指定所有数据,还有一种是指定某个ID的数据。
我们看以下的样例。
· content://contacts/people/ 这个Uri指定的就是所有的联系人数据。
· content://contacts/people/1 这个Uri指定的是ID为1的联系人的数据。
在上边两个类中用到的Uri一般由3部分组成。
· 第一部分是方案:"content://" 这部分永远不变
· 第二部分是授权:"contacts"
· 第二部分是路径:"people/","people/1"(假设没有指定ID,那么表示返回所有)。
因为URI通常比較长,并且有时候easy出错,且难以理解。所以,在Android其中定义了一些辅助类,并且定义了一些常量来取代这些长字符串的使用,比例如以下边的代码:
· Contacts.People.CONTENT_URI (联系人的URI)。
在我们的实例MyProvider中是例如以下定义的:
public static final String AUTHORITY="com.teleca.PeopleProvider";
public static final String PATH_SINGLE="people/#";
public static final String PATH_MULTIPLE="people";
public static final Uri content_URI=Uri.parse("content://"+AUTHORITY+"/"+PATH_MULTIPLE);
实例1:
文件MyProvider.java
package com.teleca.provider;
import java.util.HashMap;
import android.content.ContentProvider;
import android.content.ContentUris;
import android.content.ContentValues;
import android.content.Context;
import android.content.UriMatcher;
import android.database.Cursor;
import android.database.SQLException;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;
import android.database.sqlite.SQLiteQueryBuilder;
import android.net.Uri;
import android.text.TextUtils;
import android.util.Log;
public class MyProvider extends ContentProvider {
public static final String MIME_DIR_PREFIX="vnd.android.cursor.dir";
public static final String MIME_ITEM_PREFIX="vnd.android.cursor.item";
public static final String MIME_ITEM="vnd.msi.people";
public static final String MIME_TYPE_SINGLE=MIME_ITEM_PREFIX+"/"+MIME_ITEM;
public static final String MIME_TYPE_MULTIPLE=MIME_DIR_PREFIX+"/"+MIME_ITEM;
public static final String AUTHORITY="com.teleca.PeopleProvider";
public static final String PATH_SINGLE="people/#";
public static final String PATH_MULTIPLE="people";
public static final Uri content_URI=Uri.parse("content://"+AUTHORITY+"/"+PATH_MULTIPLE);
public static final String DEFAULT_SORT_ORDER="name DESC";
public static final String _ID="_id";
public static final String NAME="name";
public static final String PHONE="phone";
public static final String AGE="age";
public static final int PEOPLE=1;
public static final int PEOPLES=2;
private static UriMatcher URI_MATCHER;
private static HashMap<String,String> PROJECTION_MAP;
public static String DB_NAME="peopledb";
public static String DB_TABLE_NAME="people";
SQLiteDatabase db;
DBOpenHelper dbOpenHelper;
static
{
URI_MATCHER=new UriMatcher(UriMatcher.NO_MATCH);
URI_MATCHER.addURI(AUTHORITY, PATH_MULTIPLE, PEOPLES);
URI_MATCHER.addURI(AUTHORITY, PATH_SINGLE, PEOPLE);
PROJECTION_MAP=new HashMap<String,String>();
PROJECTION_MAP.put(_ID, "_id");
PROJECTION_MAP.put(NAME, "name");
PROJECTION_MAP.put(PHONE, "phone");
PROJECTION_MAP.put(AGE, "age");
}
@Override
public int delete(Uri uri, String selection, String[] selectionArgs) {
// TODO Auto-generated method stub
int count=0;
switch(URI_MATCHER.match(uri))
{
case PEOPLES:
count=db.delete(DB_TABLE_NAME, selection, selectionArgs);
break;
case PEOPLE:
String segment =uri.getPathSegments().get(1);
String where="";
if(!TextUtils.isEmpty(selection))
{
where=" AND ("+selection+")";
}
count=db.delete(DB_TABLE_NAME, "_id="+segment+where, selectionArgs);
break;
default:
throw new IllegalArgumentException("Unkonw URI"+uri);
}
getContext().getContentResolver().notifyChange(uri, null);//@2
return count;
}
@Override
public String getType(Uri uri) {
// TODO Auto-generated method stub
switch(URI_MATCHER.match(uri))
{
case PEOPLES:
return MIME_TYPE_MULTIPLE;
case PEOPLE:
return MIME_TYPE_SINGLE;
default:
throw new IllegalArgumentException("Unkown URI "+uri);
}
}
@Override
public Uri insert(Uri uri, ContentValues values) {
// TODO Auto-generated method stub
long rowId=0L;
if(URI_MATCHER.match(uri)!=PEOPLES)
{
throw new IllegalArgumentException("Unkown URI"+uri);
}
rowId=db.insert(DB_TABLE_NAME, null, values);
if(rowId>0)
{
Uri result=ContentUris.withAppendedId(content_URI, rowId);
getContext().getContentResolver().notifyChange(result, null);//@2
return result;
}
else
throw new SQLException("Failed to insert row into "+uri);
}
@Override
public boolean onCreate() {
// TODO Auto-generated method stub
dbOpenHelper=new DBOpenHelper(this.getContext(),DB_NAME,1);
db=dbOpenHelper.getWritableDatabase();
return false;
}
@Override
public Cursor query(Uri uri, String[] projection, String selection,
String[] selectionArgs, String sortOrder) {
// TODO Auto-generated method stub
SQLiteQueryBuilder queryBuilder=new SQLiteQueryBuilder();
queryBuilder.setTables(DBInfo.DB_TABLE_NAME);
queryBuilder.setProjectionMap(PROJECTION_MAP);
switch(URI_MATCHER.match(uri))
{
case PEOPLES:
break;
case PEOPLE:
queryBuilder.appendWhere("_id="+uri.getPathSegments().get(1));
break;
default:
throw new IllegalArgumentException("Unkonw URI"+uri);
}
String orderBy=null;
if(TextUtils.isEmpty(sortOrder))
{
orderBy=DEFAULT_SORT_ORDER;
}
else
orderBy=sortOrder;
Cursor c=queryBuilder.query(db, projection, selection, selectionArgs, null, null, orderBy);
c.setNotificationUri(getContext().getContentResolver(), uri);//@1
return c;
}
@Override
public int update(Uri uri, ContentValues values, String selection,
String[] selectionArgs) {
// TODO Auto-generated method stub
int count=0;
switch(URI_MATCHER.match(uri))
{
case PEOPLES:
count=db.update(DB_TABLE_NAME, values, selection, selectionArgs);
break;
case PEOPLE:
String segment =uri.getPathSegments().get(1);
String where="";
if(!TextUtils.isEmpty(selection))
{
where=" AND ("+selection+")";
}
count=db.update(DB_TABLE_NAME, values, "_id="+segment+where, selectionArgs);
break;
default:
throw new IllegalArgumentException("Unkonw URI"+uri);
}
getContext().getContentResolver().notifyChange(uri, null);//@2
return count;
}
}
class DBOpenHelper extends SQLiteOpenHelper
{
private static final String DB_CREATE="CREATE TABLE "
+DBInfo.DB_TABLE_NAME
+" (_id INTEGER PRIMARY KEY,name TEXT UNIQUE NOT NULL,"
+"phone TEXT,age INTEGER);";
final static String tag="hubin";
public DBOpenHelper(Context context,String dbName,int version)
{
super(context,dbName,null,version);
}
public void onCreate(SQLiteDatabase db)
{
try{
db.execSQL(DB_CREATE);
}
catch(SQLException e )
{
Log.e(tag,"",e);
}
}
public void onOpen(SQLiteDatabase db)
{
super.onOpen(db);
}
public void onUpgrade(SQLiteDatabase db,int oldVersion,int newVersion)
{
db.execSQL("DROP TABLE IF EXISTS "+DBInfo.DB_TABLE_NAME);
this.onCreate(db);
}
}
class DBInfo
{
public static String DB_NAME="peopledb";
public static String DB_TABLE_NAME="people";
}
注意1:c.setNotificationUri(getContext().getContentResolver(), uri);
这里是把Cursor C加入到ContentResolver的监督对象组中去。
一旦有与uri相关的变化,ContentResolver就回通知Cursor C.
可能Cursor有个私有的内部类ContentObserver的实现。ContentResolver是通过该类来通知Cursor的。
public abstract void setNotificationUri (ContentResolver cr, Uri uri)
Register to watch a content URI for changes. This can be the URI of a specific data row (for example, "content://my_provider_type/23"),
or a a generic URI for a content type.
Parameters
cr The content resolver from the caller's context. The listener attached to this resolver will be notified.
uri The content URI to watch.
注意2: getContext().getContentResolver().notifyChange(uri, null)
通知数据发生了变化。
public void notifyChange (Uri uri, ContentObserver observer)
Notify registered observers that a row was updated. To register, call registerContentObserver(). By default, CursorAdapter objects will get this notification.
Parameters
observer The observer that originated the change, may be null
这里为null的意思可能就是调用在ContentResolver中注冊的ContentObserver,反之则是调用參数指定的
文件People.java
package com.teleca.provider;
public class People {
public long id;
public String name;
public String phone;
public int age;
}
文件
Hello.java
package com.teleca.provider;
import java.util.ArrayList;
import java.util.List;
import android.app.Activity;
import android.content.ContentUris;
import android.content.ContentValues;
import android.database.Cursor;
import android.database.SQLException;
import android.net.Uri;
import android.os.Bundle;
import android.os.Handler;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
public class Hello extends Activity {
/** Called when the activity is first created. */
final static String tag="hubin";
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
Button button = (Button) findViewById(R.id.Button01);
OnClickListener listener = new OnClickListener() {
@Override
public void onClick(View v) {
cmd = CMD_ADD;
doAction();
}
};
button.setOnClickListener(listener);
Button button2 = (Button) findViewById(R.id.Button02);
OnClickListener listener2 = new OnClickListener() {
@Override
public void onClick(View v) {
cmd = CMD_UPDATE;
doAction();
}
};
button2.setOnClickListener(listener2);
Button button3 = (Button) findViewById(R.id.Button03);
OnClickListener listener3 = new OnClickListener() {
@Override
public void onClick(View v) {
cmd = CMD_QUERY;
doAction();
}
};
button3.setOnClickListener(listener3);
Button button4 = (Button) findViewById(R.id.Button04);
OnClickListener listener4 = new OnClickListener() {
@Override
public void onClick(View v) {
cmd = CMD_QUERY_ALL;
doAction();
}
};
button4.setOnClickListener(listener4);
Button button5 = (Button) findViewById(R.id.Button05);
OnClickListener listener5 = new OnClickListener() {
@Override
public void onClick(View v) {
cmd = CMD_DELETE;
doAction();
}
};
button5.setOnClickListener(listener5);
Button button6 = (Button) findViewById(R.id.Button06);
OnClickListener listener6 = new OnClickListener() {
@Override
public void onClick(View v) {
cmd = CMD_DELETE_ALL;
doAction();
}
};
button6.setOnClickListener(listener6);
mHandler = new Handler();
}
int cnt = 0;
private Handler mHandler;
int cmd = 0;
final int CMD_ADD = 1;
final int CMD_UPDATE = 2;
final int CMD_QUERY= 3;
final int CMD_QUERY_ALL = 4;
final int CMD_DELETE = 5;
final int CMD_DELETE_ALL = 6;
People people=new People();
final static String projection[]=new String[]
{"_id","name","phone","age"};
class DatabaseThread implements Runnable {
public void run() {
if (cmd == CMD_ADD) {
people.name="robin"+System.currentTimeMillis()%100;
people.phone=""+System.currentTimeMillis();
people.age=1;
ContentValues values=new ContentValues();
values.put("name", people.name);
values.put("phone", people.phone);
values.put("age", people.age);
Uri uri=getContentResolver().insert(MyProvider.content_URI, values);
people.id=ContentUris.parseId(uri);
Log.i("hubin",uri.toString());
} else if (cmd == CMD_UPDATE) {
ContentValues values=new ContentValues();
people.phone=""+System.currentTimeMillis();
values.put("phone", people.phone);
Uri uri=ContentUris.withAppendedId(MyProvider.content_URI, people.id);
getContentResolver().update(uri,values,null,null);
} else if (cmd == CMD_QUERY) {
Uri uri=ContentUris.withAppendedId(MyProvider.content_URI, people.id);
Cursor c=getContentResolver().query(uri, projection, null, null, null);
People p=get(c);
printPeople(p);
} else if (cmd == CMD_QUERY_ALL) {
Uri uri=MyProvider.content_URI;
Cursor c=getContentResolver().query(uri, projection, null, null, null);
List<People> list=getAll(c);
int total=list.size();
for(int i=0;i<total;i++)
{
printPeople(list.get(i));
}
}
else if (cmd==CMD_DELETE)
{
Uri uri=ContentUris.withAppendedId(MyProvider.content_URI, people.id);
getContentResolver().delete(uri, null, null);
}
else if (cmd==CMD_DELETE_ALL)
{
Uri uri=MyProvider.content_URI;
getContentResolver().delete(uri, null, null);
}
cnt++;
}
}
void printPeople(People p)
{
Log.i(tag, "id:"+p.id);
Log.i(tag, "name:"+p.name);
Log.i(tag,"phone:"+p.phone);
Log.i(tag,"age:"+p.age);
}
DatabaseThread dataDealer=new DatabaseThread();
void doAction() {
mHandler.post(dataDealer);
}
public People get(Cursor c)
{
People people=new People();
try{
Log.i(tag,"count:"+c.getCount());
if(c.getCount()>0)
{
c.moveToFirst();
people=new People();
people.id=c.getLong(0);
people.name=c.getString(1);
people.phone=c.getString(2);
people.age=c.getInt(3);
}
}catch(SQLException e)
{
Log.i(tag,"",e);
}
finally
{
if(c!=null&&!c.isClosed())
{
c.close();
}
}
return people;
}
public List<People> getAll(Cursor c)
{
ArrayList<People> ret=new ArrayList<People>();
try
{
int count=c.getCount();
c.moveToFirst();
People people;
for(int i=0;i<count;i++)
{
people=new People();
people.id=c.getLong(0);
people.name=c.getString(1);
people.phone=c.getString(2);
people.age=c.getInt(3);
ret.add(people);
c.moveToNext();
}
}catch(SQLException e)
{
Log.i(tag,"",e);
}
finally
{
if(c!=null&&!c.isClosed())
{
c.close();
}
}
return ret;
}
}
注意:Cursor c不用时要关掉。
文件AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.teleca.provider"
android:versionCode="1"
android:versionName="1.0">
<application android:icon="@drawable/icon" android:label="@string/app_name">
<activity android:name=".Hello"
android:label="@string/app_name">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<provider android:syncable="true" android:name="MyProvider" android:authorities="com.teleca.PeopleProvider"></provider>
</application>
<uses-sdk android:minSdkVersion="7" />
</manifest>
/>
真的须要这些权限?为什么须要呢?或许是我曾经写错了吧
ContentProvider简要的更多相关文章
- 【理论基础】ContentProvider的简要概述
对于Android应用而言,他们必须相互独立,各自运行在自己的Dalvik虚拟机实例中,如果这些Android应用之间需要实现实时的数据交换——例如我们开发了一个发送短信的程序,当发送短信时需要从联系 ...
- ContentResolver + SqliteOpenHelper + ContentProvider 理解
惭愧,现在才接触到ContentResolver的用法 这个类主要是Android用来实现应用程序之间数据共享的 一个应用程序可以将自己的数据完全暴露出去,外界更本看不到,也不用看到这个应用程序暴露的 ...
- Android中使用ContentProvider进行跨进程方法调用
原文同一时候发表在我的博客 点我进入还能看到很多其它 需求背景 近期接到这样一个需求,须要和别的 App 进行联动交互,比方下载器 App 和桌面 App 进行联动.桌面的 App 能直接显示下载器 ...
- ContentResolver,ContentProvider,ContentObserver使用记录
版权声明:本文出自汪磊的博客,转载请务必注明出处. 本篇博客只是记录一下ContentProvider的使用(这部分工作中用的比较少总是忘记),没有太深入研究.已经熟练掌握使用方式,想深入了解内部机制 ...
- android 多应用程序数据共享 ContentProvider和ContentResolver
android 没有一个可以将所有应用程序数据统一放置的地方,即两个应用程序间的数据不能共享.但ContentProvider与ContentResolver可以解决多应用程序数据共享. 我们都知 ...
- Android之ContentProvider数据存储
一.ContentProvider保存数据介绍 一个程序可以通过实现一个ContentProvider的抽象接口将自己的数据完全暴露出去,而且ContentProvider是以类似数据库中表的方式将数 ...
- Xamarin.Android之ContentProvider
一.前言 掌握了如何使用SQLiteOpenHelper之后,我们就可以进行下一步的学习.本章我们将会学习如何使用ContentProvider来将数据库方面的操作封装起来,同时它还可以供其他应用访问 ...
- ContentProvider域名替换小工具
开发项目域名想怎么换就怎么换,就是这么任性! 这是一个很有意思的小工具! 这是一个方便开发人员和测试人员的小工具!! 吐槽: 一直在做Android开发,一直总有一个问题存在:做自己公司的apk开发时 ...
- JavaScript权威设计--JavaScript函数(简要学习笔记十一)
1.函数调用的四种方式 第三种:构造函数调用 如果构造函数调用在圆括号内包含一组实参列表,先计算这些实参表达式,然后传入函数内.这和函数调用和方法调用是一致的.但如果构造函数没有形参,JavaScri ...
随机推荐
- Xcode4.5 本地化,多语言设置
网上已有很多关于ios本地化的博客和资料,由于部分原作者使用的Xcode版本较早,4.5以后的版本已不再支持该方法,后来也没有更新,因此在此写一点学习资料分享出来.废话不多说. ios本地化主 ...
- C#写PDF文件类库PDF File Writer介绍
.NET平台开源项目速览(16)C#写PDF文件类库PDF File Writer介绍 阅读目录 1.PDF File Writer基本介绍 2.一个简单的使用案例 3.资源 1年前,我在文章:这 ...
- HDU 4160 Dolls (最小路径覆盖=顶点数-最大匹配数)
Dolls Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submiss ...
- 一个带动画效果的颜色选择对话框控件AnimatedColorPickerDialog
android4.4的日历中选择日程显示颜色的时候有一个颜色选择对话框非常漂亮,模仿他的界面我实现了一个类似的对话框,而且带有动画效果. 代码的实现可讲的地方不多,主要是采用了和AlertDialog ...
- Ubuntu14.04更新源
Ubuntu14.04更新源 http://jingyan.baidu.com/article/7f41ecec1b7a2e593d095ce6.html Ubuntu源 http://wiki.ub ...
- 关于Linux路由表的route命令(转)
查看 Linux 内核路由表 使用下面的 route 命令可以查看 Linux 内核路由表. # route Destination Gateway Genmask Fl ...
- 基于visual Studio2013解决C语言竞赛题之1039移动
题目 解决代码及点评 /* 39. 有n个整数,编程序将前面的各个数依次向后移动k个位置, 最后k个数移到最前边的k个位置(见下图,其中n=8,k=3). */ # ...
- ctfmon.exe开机无法自己主动启动
打开命令提示符(開始菜单--执行--输入:cmd),输入下面命令(复制粘贴就可以): reg add HKCU\Software\Microsoft\Windows\CurrentVersion\Ru ...
- linux c: core dump
1. core dump文件系统设置 http://www.cnblogs.com/no7dw/archive/2013/02/18/2915819.html 编译时需要输入-g才会生成coredum ...
- 为什么要用BitSet
BitSet适用于一类型boolean判断,Java的BitSet在这类型判断中非常高效. 举例说明:在判断前2000万数字中素数个数的程序中,如果使用最基本的素数判断代码: package com; ...