ContentProvider

ContentProvider!

字数952  阅读1  评论0 

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内部的路径部分,CD部分称为路径片段,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应用之间的相互操作!


相互学习,共同进步!

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值