Android学习笔记_10_ContentProvider内容提供者的使用
一、使用ContentProvider共享数据
当应用继承ContentProvider类,并重写该类用于提供数据和存储数据的方法,就可以向其他应用共享其数据。以前我们学习过文件的操作模式,通过指定文件的操作模式为Context.MODE_WORLD_READABLE 或Context.MODE_WORLD_WRITEABLE同样可以对外共享数据,但数据的访问方式会因数据存储的方式而不同,如:采用xml文件对外共享数据,需要进行xml解析来读写数据;采用sharedpreferences共享数据,需要使用sharedpreferences API读写数据。而使用ContentProvider共享数据的好处是统一了数据访问方式。
当应用需要通过ContentProvider对外共享数据时,第一步需要继承ContentProvider并重写下面方法:
- package com.example.service;
- import android.content.ContentProvider;
- import android.content.ContentUris;
- import android.content.ContentValues;
- import android.content.UriMatcher;
- import android.database.Cursor;
- import android.database.sqlite.SQLiteDatabase;
- import android.net.Uri;
- public class PersonProvider extends ContentProvider {
- private DBOpenHelper helper;
- // 不匹配返回-1
- private static final UriMatcher MATCHER = new UriMatcher(
- UriMatcher.NO_MATCH);
- private static final int PERSONS = 1;// 表示多个
- private static final int PERSON = 2; // 表示一条记录
- static {
- MATCHER.addURI("com.example.providers.personprovider", "person",
- PERSONS);
- MATCHER.addURI("com.example.providers.personprovider", "person/#",
- PERSON);
- }
- // 删除
- @Override
- public int delete(Uri uri, String selection, String[] selectionArgs) {
- SQLiteDatabase db = helper.getWritableDatabase();
- int num = 0;
- switch (MATCHER.match(uri)) {
- case 1:
- num = db.delete("person", selection, selectionArgs);
- break;
- case 2:
- long id = ContentUris.parseId(uri);
- String where = "id=" + id;
- if (selection != null && !"".equals(selection.trim())) {
- where += " and " + selection;
- }
- num = db.delete("person", where, selectionArgs);
- break;
- default:
- throw new IllegalArgumentException("uri参数不正确");
- }
- return num;
- }
- @Override
- public String getType(Uri uri) {
- switch (MATCHER.match(uri)) {
- case 1:
- return "vnd.android.cursor.dir/person";
- case 2:
- return "vnd.android.cursor.item/person";
- default:
- throw new IllegalArgumentException("uri参数不正确");
- }
- }
- @Override
- public Uri insert(Uri uri, ContentValues values) {
- SQLiteDatabase db = helper.getWritableDatabase();
- switch (MATCHER.match(uri)) {
- case 1:
- long rowid = db.insert("person", "name", values);
- // content://com.example.providers.personprovider/person/1
- // Uri insertUri =
- // Uri.parse("content://com.example.providers.personprovider/person/"+rowid);
- // 在原来的uri后面加上一个rowid
- Uri insertUri = ContentUris.withAppendedId(uri, rowid);
- return insertUri;
- default:
- throw new IllegalArgumentException("uri参数不正确");
- }
- }
- @Override
- public boolean onCreate() {
- helper = new DBOpenHelper(this.getContext());
- return false;
- }
- @Override
- public Cursor query(Uri uri, String[] projection, String selection,
- String[] selectionArgs, String sortOrder) {
- SQLiteDatabase db = helper.getReadableDatabase();
- switch (MATCHER.match(uri)) {
- case 1:
- return db.query("person", projection, selection, selectionArgs,
- null, null, sortOrder);
- case 2:
- long id = ContentUris.parseId(uri);
- String where = "id=" + id;
- if (selection != null && !"".equals(selection.trim())) {
- where += " and " + selection;
- }
- return db.query("person", projection, where, selectionArgs, null,
- null, sortOrder);
- default:
- throw new IllegalArgumentException("uri参数不正确");
- }
- }
- @Override
- public int update(Uri uri, ContentValues values, String selection,
- String[] selectionArgs) {
- SQLiteDatabase db = helper.getWritableDatabase();
- int num = 0;
- switch (MATCHER.match(uri)) {
- case 1:
- num = db.update("person", values, selection, selectionArgs);
- break;
- case 2:
- long id = ContentUris.parseId(uri);
- String where = "id=" + id;
- if (selection != null && !"".equals(selection.trim())) {
- where += " and " + selection;
- }
- num = db.update("person", values, where, selectionArgs);
- break;
- default:
- throw new IllegalArgumentException("uri参数不正确");
- }
- return num;
- }
- }
第二步需要在AndroidManifest.xml使用<provider>对该ContentProvider进行配置,为了能让其他应用找到该ContentProvider , ContentProvider 采用了authorities(主机名/域名)对它进行唯一标识,你可以把 ContentProvider看作是一个网站(想想,网站也是提供数据者),authorities 就是他的域名:
- <manifest>
- <application android:icon="@drawable/icon" android:label="@string/app_name">
- <!--
- 配置内容提供者:
- name:PersonProvider表示内容提供者类名,包名+类。
- authorities: 内容提供者的唯一标识
- -->
- <provider android:name="com.example.service.PersonProvider"
- android:authorities="com.example.providers.personprovider" >
- </provider>
</application>- </manifest>
二、在另一个应用进行测试上面的代码:
需要用到ContentResolver和ContentValues两个类,前者封装了内容提供者的增删改查,后者实体类数据。
- package com.example.test;
- import android.content.ContentResolver;
- import android.content.ContentValues;
- import android.database.Cursor;
- import android.net.Uri;
- import android.test.AndroidTestCase;
- import android.util.Log;
- public class ContentProviderTest extends AndroidTestCase{
- public void testInsert(){
- Uri uri=Uri.parse("content://com.example.providers.personprovider/person");
- ContentResolver resolver = this.getContext().getContentResolver();
- ContentValues values = new ContentValues();
- values.put("name", "content");
- values.put("age", "500");
- resolver.insert(uri, values);
- }
- public void testUpdate(){
- Uri uri=Uri.parse("content://com.example.providers.personprovider/person/150");
- ContentResolver resolver = this.getContext().getContentResolver();
- ContentValues values = new ContentValues();
- values.put("name", "provider");
- values.put("age", "50");
- resolver.update(uri, values, null, null);
- }
- public void testDelete(){
- Uri uri=Uri.parse("content://com.example.providers.personprovider/person/150");
- ContentResolver resolver = this.getContext().getContentResolver();
- resolver.delete(uri, null, null);
- }
- public void testQuery(){
- Uri uri=Uri.parse("content://com.example.providers.personprovider/person");
- ContentResolver resolver = this.getContext().getContentResolver();
- Cursor cursor = resolver.query(uri, null, null, null, null);
- while (cursor.moveToNext()) {
- String name = cursor.getString(cursor.getColumnIndex("name"));
- int age = cursor.getInt(cursor.getColumnIndex("age"));
- Log.i("ContentProviderTest", " "+name+" : "+age);
- }
- }
- }
三、Uri介绍
Uri代表了要操作的数据,Uri主要包含了两部分信息:1》需要操作的ContentProvider ,2》对ContentProvider中的什么数据进行操作,一个Uri由以下几部分组成:
content://com.example.providers.personprovider/person/10,其中“com.example.providers.personprovider”指主机名或authority,“person/10”表示路径。
- ContentProvider(内容提供者)的scheme已经由Android所规定, scheme为:content://
- 主机名(或叫Authority)用于唯一标识这个ContentProvider,外部调用者可以根据这个标识来找到它。
- 路径(path)可以用来表示我们要操作的数据,路径的构建应根据业务而定,如下:
- 要操作person表中id为10的记录,可以构建这样的路径:/person/10
- 要操作person表中id为10的记录的name字段, person/10/name
- 要操作person表中的所有记录,可以构建这样的路径:/person
- 要操作xxx表中的记录,可以构建这样的路径:/xxx
- 当然要操作的数据不一定来自数据库,也可以是文件、xml或网络等其他存储方式,如下:
- 要操作xml文件中person节点下的name节点,可以构建这样的路径:/person/name
- 如果要把一个字符串转换成Uri,可以使用Uri类中的parse()方法,如下:
- Uri uri = Uri.parse("content://cn.itcast.provider.personprovider/person")
四、监听内容提供者的数据变化:
1、对提供内容提供者的应用加入改变通知:
- package cn.itcast.db;
- import cn.itcast.service.DBOpenHelper;
- import android.content.ContentProvider;
- import android.content.ContentUris;
- import android.content.ContentValues;
- import android.content.UriMatcher;
- import android.database.Cursor;
- import android.database.sqlite.SQLiteDatabase;
- import android.net.Uri;
- public class PersonContentProvider extends ContentProvider {
- private static UriMatcher matcher = new UriMatcher(UriMatcher.NO_MATCH);
- private static final int PERSONS = 1;
- private static final int PERSON = 2;
- private DBOpenHelper dbOpenHelper;
- static{
- matcher.addURI("cn.itcast.providers.personprovider", "person", PERSONS);
- matcher.addURI("cn.itcast.providers.personprovider", "person/#", PERSON);
- }
- @Override
- public boolean onCreate() {
- dbOpenHelper = new DBOpenHelper(this.getContext());
- return true;
- }
- @Override
- public int delete(Uri uri, String selection, String[] selectionArgs) {
- SQLiteDatabase db = dbOpenHelper.getWritableDatabase();
- int num = 0 ;//已经删除的记录数量
- switch (matcher.match(uri)) {
- case PERSONS:
- num = db.delete("person", selection, selectionArgs);
- break;
- case PERSON:
- long id = ContentUris.parseId(uri);
- String where = "personid="+ id;
- if(selection!=null && !"".equals(selection)){ // personid=12 and name=?
- where = where + " and "+ selection;
- }
- num = db.delete("person", where, selectionArgs);
- break;
- default:
- throw new IllegalArgumentException("Unkown Uri:"+ uri);
- }
- getContext().getContentResolver().notifyChange(uri, null);
- return num;
- }
- @Override
- public String getType(Uri uri) {//返回当前操作的数据类型
- switch (matcher.match(uri)) {
- case PERSONS://操作的是集合类型数据
- return "vnd.android.cursor.dir/person";
- case PERSON:
- return "vnd.android.cursor.item/person";
- default:
- throw new IllegalArgumentException("Unkown Uri:"+ uri);
- }
- }
- @Override
- public Uri insert(Uri uri, ContentValues values) {
- SQLiteDatabase db = dbOpenHelper.getWritableDatabase();
- long id = 0 ;
- switch (matcher.match(uri)) {
- case PERSONS:
- id = db.insert("person", "personid", values);//得到记录的id
- getContext().getContentResolver().notifyChange(uri, null);
- return ContentUris.withAppendedId(uri, id);//返回代表新增记录的Uri
- case PERSON:
- id = db.insert("person", "personid", values);//得到记录的id
- String strUri = uri.toString();
- Uri personUri = Uri.parse(strUri.substring(0, strUri.lastIndexOf("/")));
- getContext().getContentResolver().notifyChange(personUri, null);
- return ContentUris.withAppendedId(personUri, id);
- default:
- throw new IllegalArgumentException("Unkown Uri:"+ uri);
- }
- }
- @Override
- public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {
- SQLiteDatabase db = dbOpenHelper.getReadableDatabase();
- switch (matcher.match(uri)) {
- case PERSONS:
- return db.query("person", projection, selection, selectionArgs, null, null, sortOrder);
- case PERSON:
- long id = ContentUris.parseId(uri);
- String where = "personid="+ id;
- if(selection!=null && !"".equals(selection)){ // personid=12 and name=?
- where = where + " and "+ selection;
- }
- return db.query("person", projection, where, selectionArgs, null, null, sortOrder);
- default:
- throw new IllegalArgumentException("Unkown Uri:"+ uri);
- }
- }
- @Override
- public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {
- SQLiteDatabase db = dbOpenHelper.getWritableDatabase();
- int num = 0 ;//已经修改的记录数量
- switch (matcher.match(uri)) {
- case PERSONS:
- num = db.update("person", values, selection, selectionArgs);
- break;
- case PERSON:
- long id = ContentUris.parseId(uri);
- String where = "personid="+ id;
- if(selection!=null && !"".equals(selection)){
- where = where + " and "+ selection;
- }
- num = db.update("person", values, where, selectionArgs);
- break;
- default:
- throw new IllegalArgumentException("Unkown Uri:"+ uri);
- }
- getContext().getContentResolver().notifyChange(uri, null);//通知数据发生变化
- return num;
- }
- }
2、注册要通知改变的uri,启动B应用:
3、执行添加方法:
Android学习笔记_10_ContentProvider内容提供者的使用的更多相关文章
- 【转】Pro Android学习笔记(七):了解Content Provider(下上)
我们通过一个Content Provider小例子进行详细说明.数据源是一个SQLite数据库,名字为books.db,该数据库只含有一个表格,名字为books.表格中含有name,isbn,auth ...
- 【转】Pro Android学习笔记(五):了解Content Provider(上)
Content Provider是抽象数据封装和数据访问机制,例如SQLite是Android设备带有的数据源,可以封装到一个content provider中.要通过content provider ...
- Android 学习笔记之Volley(七)实现Json数据加载和解析...
学习内容: 1.使用Volley实现异步加载Json数据... Volley的第二大请求就是通过发送请求异步实现Json数据信息的加载,加载Json数据有两种方式,一种是通过获取Json对象,然后 ...
- Android组件系列----ContentProvider内容提供者
[声明] 欢迎转载,但请保留文章原始出处→_→ 生命壹号:http://www.cnblogs.com/smyhvae/ 文章来源:http://www.cnblogs.com/smyhvae/p/4 ...
- Android学习笔记之JSON数据解析
转载:Android学习笔记44:JSON数据解析 JSON(JavaScript Object Notation)是一种轻量级的数据交换格式,采用完全独立于语言的文本格式,为Web应用开发提供了一种 ...
- udacity android 学习笔记: lesson 4 part b
udacity android 学习笔记: lesson 4 part b 作者:干货店打杂的 /titer1 /Archimedes 出处:https://code.csdn.net/titer1 ...
- Android学习笔记36:使用SQLite方式存储数据
在Android中一共提供了5种数据存储方式,分别为: (1)Files:通过FileInputStream和FileOutputStream对文件进行操作.具体使用方法可以参阅博文<Andro ...
- Pro Android学习笔记 ActionBar(1):Home图标区
Pro Android学习笔记(四八):ActionBar(1):Home图标区 2013年03月10日 ⁄ 综合 ⁄ 共 3256字 ⁄ 字号 小 中 大 ⁄ 评论关闭 ActionBar在A ...
- 【转】 Pro Android学习笔记(六七):HTTP服务(1):HTTP GET
目录(?)[-] HTTP GET小例子 简单小例子 出现异常NetworkOnMainThreadException 通过StrictMode进行处理 URL带键值对 Andriod应用可利用ser ...
随机推荐
- word 快捷键
Ctrl+shift+F9 清除word文档中的超链接
- 02-使用注解配置spring
1 准备工作 1.导包 4+2+spring-aop[新版本需要导入 spring-aop 包] 2.为主配置文件引入新的命名空间(约束) [context] 3.开启使用注解代理配置文件 4.在类中 ...
- TOJ 1023 Taxi Cab Scheme
Description Running a taxi station is not all that simple. Apart from the obvious demand for a centr ...
- FZU 2216——The Longest Straight——————【二分、枚举】
Problem 2216 The Longest Straight Accept: 17 Submit: 39Time Limit: 1000 mSec Memory Limit : 32 ...
- 【linux相识相知】独立硬盘冗余阵列-RAID
独立硬盘冗余阵列(RAID,Redundant Array of Independant Disks),旧称为廉价磁盘冗余阵列(Redundant Array of Inexpensive Disks ...
- 在浏览器中对访问的网页中的cookie添加和修改
做权限相关的东西,使用到了cookie,关于它的安全性,cookie在浏览器中,通过插件是可以对其进行修改的,如下: 1.FireFox 安装Edit This Cookie 插件,之后点击插件图标即 ...
- 虚拟环境--pipenv
1.安装pipenv,这个工具属于python3 升级pip : pip3 install pipenv 2.在项目中创建虚拟环境 3.激活虚拟环境,进入虚拟环境 进入虚拟环境之前: pipenv s ...
- Trim a Binary Search Tree
Given a binary search tree and the lowest and highest boundaries as L and R, trim the tree so that a ...
- python高阶函数sorted
原文 排序也是在程序中经常用到的算法.无论使用冒泡排序还是快速排序,排序的核心是比较两个元素的大小.如果是数字,我们可以直接比较,但如果是字符串或者两个dict呢?直接比较数学上的大小是没有意义的,因 ...
- Jquery系列:textarea常用操作
1.textarea内容的读取与设置 读textarea文本值可以用name和id.而写入文本值只能用id. <textarea name="content" id=&quo ...