ContentProvider是Android四大组件之一。它主要的作用是:实现各个应用程序之间的(跨应用)数据共享,比如联系人应用中就使用了ContentProvider,你在自己的应用中可以读取和修改联系人的数据,不过需要获得相应的权限。
但是ContentProvider也只是起到中间人的作用,真正实现应用间数据传输的用SQLite。
和ContentProvider紧密祥光的就是URI!
Uri是统一标识符,代表要操作的数据,标识每一个ContentProvider,当需要的时候就通过Uri来查找对应想要的ContentProvider。从而获取和修改等其他操作。
Android中的Uri的格式也是一下的形式,主要分为三个部分:
A 部分:表示是一个Android内容URI,说明由ContentProvider控制数据,该部分是固定形式,不可更改的。
�B 部分:是URI的授权部分,是唯一标识符,用来定位ContentProvider。
格式一般是自定义ContentProvider类的完全限定名称,注册时需要用到,
如:com.alexzhou.provider.NoteProvider
�C 部分和D部分:是每个ContentProvider内部的路径部分,C和D部分称为路径片段,C部分指向一个对象集合,
一般用表的名字,如:/notes表示一个笔记集合;D部分指向特定的记录,如:/notes/1表示id为1的笔记,
如果没有指定D部分,则返回全部记录。
ContentProvider可以理解为一个Android应用对外开放的接口,只要是符合它所定义的Uri格式的请求,均可以正常访问执行操作。
其他的Android应用可以使用ContentResolver对象通过与ContentProvider同名的方法请求执行,被执行的就是ContentProvider中的同名方法。
所以ContentProvider很多对外可以访问的方法,在ContentResolver中均有同名的方法,是一一对应的!
ContentProvider必须要实现的几个方法:
1、onCreate():初始化提供者。
@Override
public boolean onCreate() {
// TODO Auto-generated method stub
Log.e(TAG, "onCreate");
helper = new MySqliteHelper(getContext());
matcher = new UriMatcher(UriMatcher.NO_MATCH);
matcher.addURI(authority, tableName, STUDENT);
matcher.addURI(authority, teachName, TEACHER);
return true;
}```
2、insert(Uri, ContentValues):插入一条数据。
3、query(Uri, String[], String, String[], String):查询数据,返回一个数据Cursor对象。(根据括号里的参数查询数据,后面类型的填null代表所有的数据都要查询)
@Override
public Cursor query(Uri uri, String[] projection, String selection,
String[] selectionArgs, String sortOrder) {
// TODO Auto-generated method stub
int code = matcher.match(uri);
db = helper.getReadableDatabase();
switch (code) {
case TEACHER:
Log.e(TAG, "teacher_query");
Cursor teachCursor = db.query(teachName, projection, selection,
selectionArgs, null, null, null);
return teachCursor;
case STUDENT:
Log.e(TAG, "teacher_query");
Cursor cursor = db.query(tableName, projection, selection,
selectionArgs, null, null, null);
return cursor;
}
return null;
}```
4、delete(Uri, String, String[]):根据条件删除数据。
@Override
public int delete(Uri uri, String selection, String[] selectionArgs) {
// TODO Auto-generated method stub
int code=matcher.match(uri);
db = helper.getReadableDatabase();
switch(code){
case TEACHER:
Log.e(TAG, "teacher_delete");
db.delete(teachName, selection, selectionArgs);
break;
case STUDENT:
Log.e(TAG, "student_delete");
db.delete(tableName, selection, selectionArgs);
break;
}
return 0;
}```
在ContentProvider的增删改查操作中,均会传递一个Uri对象,通过这个对象来匹配对应的请求。那么如何确定一个Uri执行哪项操作呢?
需要用到一个UriMatcher对象,这个对象用来帮助内容提供者匹配Uri。它所提供的方法非常简单,仅有两个:
void addURI(String authority,String path,int code):添加一个Uri匹配项,authority为AndroidManifest.xml中注册的ContentProvider中的authority属性;path为一个路径,可以设置通配符,#表示任意数字,*表示任意字符;code为自定义的一个Uri代码。
int match(Uri uri):匹配传递的Uri,返回addURI()传递的code参数。
在创建好一个ContentProvider之后,还需要在AndroidManifest.xml文件中对ContentProvider进行配置,使用一个<provider.../>节点,一般只需要设置两个属性即可访问,一些额外的属性就是为了设置访问权限而存在的,后面会详细讲解:
android:name:provider的响应类。
android:authorities:Provider的唯一标识,用于Uri匹配,一般为ContentProvider类的全名。
ContentProvider也是四大组件之一所以需要在Manifest中注册;
在ContentProvider之前要创建一个类(MySqliteHelper)来继承SQLiteOpenHelper这个类,其中要调用一个构造方法和两个方法(一个创建的方法,一个更新版本的方法)。在创建的方法里要创建一个表格要用用Sql语句来完成
@Override
public void onCreate(SQLiteDatabase db) {
// TODO Auto-generated method stub
db.execSQL("CREATE TABLE Student(id integer primary key autoincrement,name varchar(20),age varchar(10))");
db.execSQL("CREATE TABLE Teacher(id integer primary key autoincrement,tname varchar(20),tage varchar(10))");
}```
都写好了之后,就在另一个程序内加上点击的事件---增删改查的点击方法,来操作ContentProvider内写好的对应的增删改查
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
switch(v.getId()){
case R.id.but_insert:
ContentValues values=new ContentValues();
values.put("tname", "老师");
values.put("tage", "23");
ContentValues values1=new ContentValues();
values1.put("tname", "老师师");
values1.put("tage", "123");
resolver.insert(teachUri, values);
resolver.insert(teachUri, values1);
break;
case R.id.but_update:
ContentValues updateValues=new ContentValues();
updateValues.put("tname", "李四");
updateValues.put("tage", "32");
resolver.update(teachUri, updateValues, "tname=?", new String[]{"老师"});
break;
case R.id.but_query:
List<String> list=new ArrayList<String>();
Cursor cursor=resolver.query(teachUri, null, null, null, null);
while(cursor.moveToNext()){
int nameIndex=cursor.getColumnIndex("tname");
int ageIndex=cursor.getColumnIndex("tage");
String nameStr=cursor.getString(nameIndex);
String ageStr=cursor.getString(ageIndex);
list.add(nameStr+ageStr);
}
ArrayAdapter adapter=new ArrayAdapter(this, android.R.layout.simple_list_item_1, list);
lv.setAdapter(adapter);
break;
case R.id.but_delete:
resolver.delete(teachUri, "tname=?", new String[]{"李四"});
break;
}
}```
准备工作就是生命一个Uri、ContentResolver。
String uriStr="content://com.example.contentprovider.MyProvider/Student";
uri=Uri.parse(uriStr);
将一个字符串转化成一个Uri格式的类型
resolver=getContentResolver();
获取resolver的实例
通过上述操作可以实现ContentProvider应用之间的相互操作!
-----------------------------------------------------------------------------
相互学习,共同进步!
------------------------------------------------------------------------------