美文网首页
ContentProvider!

ContentProvider!

作者: Judy_pass | 来源:发表于2016-12-21 19:49 被阅读69次

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部分,则返回全部记录。
Paste_Image.png

ContentProvider可以理解为一个Android应用对外开放的接口,只要是符合它所定义的Uri格式的请求,均可以正常访问执行操作。
其他的Android应用可以使用ContentResolver对象通过与ContentProvider同名的方法请求执行,被执行的就是ContentProvider中的同名方法。
所以ContentProvider很多对外可以访问的方法,在ContentResolver中均有同名的方法,是一一对应的!

Paste_Image.png

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应用之间的相互操作!

-----------------------------------------------------------------------------
相互学习,共同进步!
------------------------------------------------------------------------------













相关文章

网友评论

      本文标题:ContentProvider!

      本文链接:https://www.haomeiwen.com/subject/jqrzmttx.html