Android学习总结——Content Provider
原文地址:http://www.cnblogs.com/bravestarrhu/archive/2012/05/02/2479461.html
Content Provider内容提供者 :
android平台提供了Content Provider使一个应用程序的指定数据集提供给其他应用程序。这些数据可以存储在文件系统中、在一个SQLite数据库、或以任何其他合理的方式,其他应用可以通过ContentResolver类(见ContentProviderAccessApp例子)从该内容提供者中获取或存入数据.(相当于在应用外包了一层壳),只有需要在多个应用程序间共享数据是才需要内容提供者。例如,通讯录数据被多个应用程序使用,且必须存储在一个内容提供者中
它的好处:统一数据访问方式。
android系统自带的内容提供者(顶级的表示数据库名,非顶级的都是表名)这些内容提供者在SDK文档的android.provider Java包中都有介绍。见:http://developer.android.com/reference/android/provider/package-summary.html
├────Browser
├────CallLog
├────Contacts
│ ├────Groups
│ ├────People
│ ├────Phones
│ └────Photos
├────Images
│ └────Thumbnails
├────MediaStore
│ ├────Albums
│ ├────Artists
│ ├────Audio
│ ├────Genres
│ └────Playlists
├────Settings
└────Video
CallLog:地址和接收到的电话信息
Contact.People.Phones:存储电话号码
Setting.System:系统设置和偏好设置
使用Content Provider对外共享数据的步骤
1>继承ContentProvider类并根据需求重写以下方法:
public boolean onCreate();//处理初始化操作
/**
* 插入数据到内容提供者(允许其他应用向你的应用中插入数据时重写)
* @param uri
* @param initialValues 插入的数据
* @return
*/
public Uri insert(Uri uri, ContentValues initialValues);
/**
* 从内容提供者中删除数据(允许其他应用删除你应用的数据时重写)
* @param uri
* @param selection 条件语句
* @param selectionArgs 参数
* @return
*/
public int delete(Uri uri, String selection, String[] selectionArgs);
/**
* 更新内容提供者已存在的数据(允许其他应用更新你应用的数据时重写)
* @param uri
* @param values 更新的数据
* @param selection 条件语句
* @param selectionArgs 参数
* @return
*/
public int update(Uri uri, ContentValues values, String selection,
String[] selectionArgs);
/**
* 返回数据给调用者(允许其他应用从你的应用中获取数据时重写)
* @param uri
* @param projection 列名
* @param selection 条件语句
* @param selectionArgs 参数
* @param sortOrder 排序
* @return
*/
public Cursor query(Uri uri, String[] projection, String selection,
String[] selectionArgs, String sortOrder) ;
/**
* 用于返回当前Uri所代表数据的MIME类型
* 如果操作的数据为集合类型(多条数据),那么返回的类型字符串应该为vnd.android.cursor.dir/开头
* 例如要得到所有person记录的Uri为content://com.bravestarr.provider.personprovider/person,
* 那么返回的MIME类型字符串应该为"vnd.android.cursor.dir/person"
* 如果操作的数据为单一数据,那么返回的类型字符串应该为vnd.android.cursor.item/开头
* 例如要得到id为10的person记录的Uri为content://com.bravestarr.provider.personprovider/person/10,
* 那么返回的MIME类型字符串应该为"vnd.android.cursor.item/person"
* @param uri
*/
public String getType(Uri uri)
这些方法中的Uri参数,得到后需要进行解析然后做对应处理,Uri表示要操作的数据,包含两部分信息:
1.需要操作的contentprovider
2.对contentprovider中的什么数据进行操作,一个Uri格式:结构头://authorities(域名)/路径(要操作的数据,根据业务而定)
content://com.bravestarr.provider.personprovider/person/10
说明:contentprovider的结构头已经由android规定为content://
authorities用于唯一标识这个contentprovider程序,外部调用者可以根据这个找到他
路径表示我们要操作的数据,路径的构建根据业务而定.路径格式如下:
要操作person表行号为10的记录,可以这样构建/person/10
要操作person表的所有记录,可以这样构建/person
2>在AndroidManifest.xml中使用<provider>对ContentProvider进行配置注册(内容提供者注册它自己就像网站注册域名),ContentProvider采用authoritie(原意授权,可理解为域名)作为唯一标识,方便其他应用能找到
<application
android:icon="@drawable/ic_launcher"
android:label="@string/app_name" >
<!-- authorities属性命名建议:公司名.provider.SomeProvider-->
<provider android:name=".PersonProvider" android:authorities="com.bravestarr.provider.personprovider"/>
...
</application>
添加内容,便于复习:
ContentProvider的getType()方法:mime类型,在返回的记录有多条时,以vnd.android.dir/开头,返回的记录只有一条时,以vnd.android.cursor.item/开头
何为Uri?
-Uri是指通用资源标志符
content://com.imooc.provider/music/#
content://---前缀表明数据受控于一个内容提供者.它从不修改,也就是schema
com.imooc.provider---是指在AndroidMainfest.xml中我们注册的provider中的authorities属性所对应的唯一的
/music---具体操作于哪个条目
/#---具体指定到哪个条目下的那条记录(#标识通配符)
图片来源:慕课网
Demo:
package com.example.contentproviderdemo;
import android.content.ContentResolver;
import android.database.Cursor;
import android.provider.ContactsContract.CommonDataKinds.Email;
import android.provider.ContactsContract.CommonDataKinds.Phone;
import android.provider.ContactsContract.Contacts;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ContentResolver resolver=getContentResolver();
Cursor cs=resolver.query(Contacts.CONTENT_URI,new String[]{Contacts._ID,Contacts.DISPLAY_NAME},null,null,null);
if(cs!=null){
while (cs.moveToNext()){
int id=cs.getInt(cs.getColumnIndex("_ID"));
String name=cs.getString(cs.getColumnIndex(Contacts.DISPLAY_NAME));
Log.i("info","_id: "+id);
Log.i("info","_name: "+name);
// 根据联系人ID查询出联系人的电话号码
/**
* 查电话号码,需要先查出人名和对应的id,然后才能根据人的id把所有人的号码查出来再得到一个游标对象进行处理(游标套游标),
* 例子如下:下面的id就是联系人的id,查询联系人的id是查询的联系人的那个表。而查询电话号码是另一个表了,
* 但是通过联系人id和联系人表连在一起的。
*/
Cursor cs1=resolver.query(Phone.CONTENT_URI,new String[]{Phone.NUMBER,Phone.TYPE},Phone.CONTACT_ID+"="+id,null,null);
if(cs1!=null){
while (cs1.moveToNext()){
int type=cs1.getInt(cs1.getColumnIndex(Phone.TYPE));
if(type==Phone.TYPE_HOME){
Log.i("info","家庭电话:"+cs1.getString(cs1.getColumnIndex(Phone.NUMBER)));
}else if(type==Phone.TYPE_MOBILE){
Log.i("info","手机:"+cs1.getString(cs1.getColumnIndex(Phone.NUMBER)));
}
}
}
cs1.close();
//根据联系人的ID查询联系人的邮箱地址
Cursor cs2=resolver.query(Email.CONTENT_URI, new String[]{Email.DATA,Email.TYPE}, Email.CONTACT_ID+"="+id, null, null);
if(cs2!=null){
while(cs2.moveToNext()){
int type=cs2.getInt(cs2.getColumnIndex(Email.TYPE));
String data=cs2.getString(cs2.getColumnIndex(Email.DATA));
if(type==Email.TYPE_WORK){
Log.i("info", "工作邮箱:"+data);
}
}
}
cs2.close();
}
}
cs.close();
}
}
增加联系人: