【攻克Android (45)】四大组件之 ContentProvider

[b][size=large]本文围绕以下两个部分展开:[/size][/b]

[b][size=large]一、ContentProvider[/size][/b]
[b][size=large]案例一:获得手机通讯录中的所有联系人[/size][/b]


[b][size=large]一、ContentProvider[/size][/b]

[size=medium][b]1. 使用ContentProvider共享数据与其他方式的区别[/b][/size]

[size=medium]通过指定文件的操作模式为Context.MODE_WORLD_READABLE
或Context.MODE_WORLD_WRITEABLE同样可以对外共享数据,但数据的访问方式会因数据存储的方式而不同。[/size]

[size=medium]如:采用xml文件对外共享数据,需要进行xml解析来读写数据;[/size]

[size=medium]采用sharedpreferences共享数据,需要使用sharedpreferences API读写数据。[/size]

[size=medium]而使用ContentProvider共享数据的好处是[b]统一了数据访问方式[/b]。当应用继承ContentProvider类,并重写该类用于提供数据和存储数据的方法,就可以向其他应用共享其数据。[/size]


[size=medium][b]2. Uri介绍[/b][/size]

[size=medium][b](1)Uri代表了要操作的数据。[/b][/size]


[size=medium][b](2)Uri主要包含了两部分信息:[/b][/size]

[size=medium][b]A.[/b] 需要操作的ContentProvider。[/size]

[size=medium][b]B.[/b] 对ContentProvider中的什么数据进行操作。[/size]


[size=medium][b](3)一个Uri由以下几部分组成:[/b][/size]

[size=medium][b][color=red]content://com.xiangdonglee.provider.personprovider/person/10[/color][/b][/size]

[size=medium][b]1)schema。[/b]ContentProvider的scheme已经由Android所规定,scheme为:content://[/size]

[size=medium][b]2)Authority(或叫主机名)。[/b]用于唯一标识这个ContentProvider,外部调用者可以根据这个标识来找到它。Authority为:com.xiangdonglee.provider.personprovider[/size]

[size=medium][b]3)path(路径)。[/b]path为:person/10(或person)[/size]

[size=medium]path可以用来表示我们要操作的数据,路径的构建应根据业务而定,如下:[/size]

[size=medium]A. 要操作person表中id为10的记录,可以构建这样的路径:/person/10[/size]

[size=medium]B. 要操作person表中id为10的记录的name字段, person/10/name[/size]

[size=medium]C. 要操作person表中的所有记录,可以构建这样的路径:/person[/size]

[size=medium]D. 要操作xxx表中的记录,可以构建这样的路径:/xxx[/size]

[size=medium]当然要操作的数据不一定来自数据库,也可以是文件、xml或网络等其他存储方式,如下:[/size]

[size=medium]E. 要操作xml文件中person节点下的name节点,可以构建这样的路径:/person/name[/size]

[size=medium]如果要把一个字符串转换成Uri,可以使用Uri类中的parse()方法,如下:[/size]

[size=medium]Uri uri = Uri.parse("content://cn.xiangdonglee.provider.personprovider/person")[/size]

[size=medium][b]4)id。[/b]id为:10[/size]


[size=medium][b]3. 操作Uri的工具类[/b][/size]

[size=medium]因为Uri代表了要操作的数据,所以我们经常需要解析Uri,并从Uri中获取数据。Android系统提供了两个用于操作Uri的工具类,分别为UriMatcher 和 ContentUris 。掌握它们的使用,会便于我们的开发工作。[/size]

[size=medium][b](1)UriMatcher类[/b][/size]

[size=medium][b]1)[/b]UriMatcher类用于匹配Uri。[/size]

[size=medium][b]2)[/b]UriMatcher类用法如下:[/size]

[size=medium][b]A. [/b]把你需要匹配Uri路径全部给注册上。[/size]

    //常量UriMatcher.NO_MATCH表示不匹配任何路径的返回码
UriMatcher sMatcher = new UriMatcher(UriMatcher.NO_MATCH);
//如果match()方法匹配content://cn.xiangdonglee.provider.personprovider/person路径,返回匹配码为1
//添加需要匹配uri,如果匹配就会返回匹配码
sMatcher.addURI(“cn.xiangdonglee.provider.personprovider”, “person”, 1);
//如果match()方法匹配content://cn.xiangdonglee.provider.personprovider/person/230路径,返回匹配码为2
//#号为通配符
sMatcher.addURI(“cn.xiangdonglee.provider.personprovider”, “person/#”, 2);
switch (sMatcher.match(Uri.parse("content://cn.xiangdonglee.provider.personprovider/person/10"))) {
case 1:
break;
case 2:
break;
default:
//不匹配
break;
}


[size=medium][b]B. [/b]注册完需要匹配的Uri后,就可以使用sMatcher.match(uri)方法对输入的Uri进行匹配。[/size]

[size=medium]如果匹配就返回匹配码,匹配码是调用addURI()方法传入的第三个参数。假设匹配content://cn.xiangdonglee.provider.personprovider/person路径,返回的匹配码为1。[/size]


[size=medium][b](2)ContentUris类[/b][/size]

[size=medium][b]1)[/b]ContentUris类用于获取Uri路径后面的ID部分。[/size]

[size=medium][b]2)[/b]它有两个比较实用的方法:[/size]

[size=medium][b]A. [/b]withAppendedId(uri, id):用于为路径加上ID部分。[/size]

    Uri uri = Uri.parse("content://cn.xiangdonglee.provider.personprovider/person");
Uri resultUri = ContentUris.withAppendedId(uri, 10);
//生成后的Uri为:content://cn.xiangdonglee.provider.personprovider/person/10


[size=medium][b]B. [/b]parseId(uri):用于从路径中获取ID部分。[/size]

    Uri uri = Uri.parse("content://cn.xiangdonglee.provider.personprovider/person/10");
long personid = ContentUris.parseId(uri);
//获取的结果为:10



[size=medium][b]4. 通过ContentProvider对外共享数据的步骤:[/b][/size]

[size=medium][b](1)需要继承ContentProvider类并重写下面方法:[/b][/size]

    public class PersonContentProvider extends ContentProvider{
// (1)
public boolean onCreate(){
// 该方法在ContentProvider创建后就会被调用。
// Android开机后, ContentProvider在其它应用第一次访问它时才会被创建。
}

// (2)
public Uri insert(Uri uri, ContentValues values){
// 该方法用于供外部应用往ContentProvider添加数据。
}

// (3)
public int delete(Uri uri, String selection, String[] selectionArgs){
// 该方法用于供外部应用从ContentProvider删除数据。
}

// (4)
public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs){
// 该方法用于供外部应用更新ContentProvider中的数据。
}

// (5)
public Cursor query(Uri uri, String[] projection,
String selection, String[] selectionArgs,
String sortOrder){
// 该方法用于供外部应用从ContentProvider中获取数据。
}

// (6)
public String getType(Uri uri){
// 该方法用于返回当前Url所代表数据的MIME类型。

/*
如果操作的数据属于集合类型,
那么MIME类型字符串应该以vnd.android.cursor.dir/开头,

例如:要得到所有person记录的Uri为
content://cn.xiangdonglee.provider.personprovider/person,
那么返回的MIME类型字符串应该为:“vnd.android.cursor.dir/person”。
*/

/*
如果要操作的数据属于非集合类型数据,
那么MIME类型字符串应该以vnd.android.cursor.item/开头,
例如:得到id为10的person记录,Uri为
content://cn.xiangdonglee.provider.personprovider/person/10,
那么返回的MIME类型字符串应该为:“vnd.android.cursor.item/person”。
*/
}
}


[size=medium][b](2)需要在AndroidManifest.xml使用<provider>对该ContentProvider进行配置。[/b][/size]

<manifest .... >
<application
android:icon="@drawable/icon"
android:label="@string/app_name">

<provider
android:name=".PersonContentProvider"
android:authorities="com.xiangdonglee.providers.personprovider"/>
</application>
</manifest>


[size=medium]为了能让其他应用找到该ContentProvider,ContentProvider 采用了authorities(主机名/域名)对它进行唯一标识。你可以把 ContentProvider看作是一个网站(想想,网站也是提供数据者),authorities 就是他的域名。[/size]

[size=medium]注意:一旦应用继承了ContentProvider类,后面我们就会把这个应用称为ContentProvider(内容提供者)。[/size]


[size=medium][b]5. 使用ContentResolver操作ContentProvider中的数据:[/b][/size]

[size=medium]当外部应用需要对ContentProvider中的数据进行添加、删除、修改和查询操作时,可以使用 ContentResolver 类来完成。[/size]

[size=medium]要获取ContentResolver 对象,可以使用Activity提供的[b]getContentResolver()[/b]方法。[/size]


[size=medium][b](1)ContentResolver 类提供了与ContentProvider类相同签名的四个方法:[/b][/size]

    // (1)
public Uri insert(Uri uri, ContentValues values){
// 该方法用于往ContentProvider添加数据。
}

// (2)
public int delete(Uri uri, String selection, String[] selectionArgs){
// 该方法用于从ContentProvider删除数据。
}

// (3)
public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs){
// 该方法用于更新ContentProvider中的数据。
}

// (4)
public Cursor query(Uri uri, String[] projection,
String selection, String[] selectionArgs,
String sortOrder){
// 该方法用于从ContentProvider中获取数据。
}


[size=medium]这些方法的第一个参数为Uri,代表要操作的ContentProvider和对其中的什么数据进行操作。[/size]

[size=medium]假设给定的是:Uri.parse(“content://cn.xiangdonglee.providers.personprovider/person/10”),那么将会对主机名为cn.xiangdonglee.providers.personprovider的ContentProvider进行操作,操作的数据为person表中id为10的记录。[/size]


[size=medium][b](2)使用ContentResolver对ContentProvider中的数据进行添加、删除、修改和查询操作:[/b][/size]

    ContentResolver resolver =  getContentResolver();
Uri uri = Uri.parse("content://cn.xiangdonglee.provider.personprovider/person");

// 添加一条记录
ContentValues values = new ContentValues();
values.put("name", "xiangdonglee");
values.put("age", 25);
resolver.insert(uri, values);

// 获取person表中所有记录
Cursor cursor = resolver.query(uri, null, null, null, "personid desc");
while(cursor.moveToNext()){
Log.i("ContentTest", "personid="+ cursor.getInt(0)+ ",name="+ cursor.getString(1));
}

// 把id为1的记录的name字段值更新为liming
ContentValues updateValues = new ContentValues();
updateValues.put("name", "liming");
Uri updateIdUri = ContentUris.withAppendedId(uri, 2);
resolver.update(updateIdUri, updateValues, null, null);

// 删除id为2的记录
Uri deleteIdUri = ContentUris.withAppendedId(uri, 2);
resolver.delete(deleteIdUri, null, null);



[b][size=large]案例一:获得手机通讯录中的所有联系人[/size][/b]

[align=center][img]http://dl2.iteye.com/upload/attachment/0111/0522/0db0a8ad-bf55-3759-94c6-e8bc12d1338c.png[/img][/align]

[align=center][img]http://dl2.iteye.com/upload/attachment/0111/0524/93e08894-60ee-35ab-8f5f-b28f333605df.png[/img][/align]

[size=medium][b]1. AndroidManifest.xml。授予读写手机联系人的权限。[/b][/size]

    <uses-permission android:name="android.permission.READ_CONTACTS" />
<uses-permission android:name="android.permission.WRITE_CONTACTS" />


[size=medium][b]2. activity_main.xml。在ScrollView中写一个TextView,用于显示手机中的所有通讯录信息,并可以滚动查看。[/b][/size]

[align=center][img]http://dl2.iteye.com/upload/attachment/0111/0532/06651fdb-2cb6-3102-816b-4fc917cafb5e.png[/img][/align]

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context=".MainActivity">

<ScrollView
android:layout_width="match_parent"
android:layout_height="match_parent">

<TextView
android:id="@+id/tvContact"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/hello_world" />
</ScrollView>

</RelativeLayout>


[size=medium][b]3. MainActivity。通过ContentResolver获得手机通讯录中的所有联系人信息。[/b][/size]

package com.android.mycontact;

import android.app.Activity;
import android.content.ContentResolver;
import android.database.Cursor;
import android.net.Uri;
import android.os.Bundle;
import android.provider.ContactsContract;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.TextView;

public class MainActivity extends Activity {
private TextView tvContact;

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);

tvContact = (TextView) findViewById(R.id.tvContact);

getAllContacts();
}

/**
* 获得手机通讯录中的所有联系人
*/
private void getAllContacts() {
String text = null;
// 获取通讯录中联系人的路径(网址)
// content://com.android.contacts/contacts
Uri uri = ContactsContract.Contacts.CONTENT_URI;
// 实例化 内容访问者
ContentResolver resolver = this.getContentResolver();
// 内容访问者通过指定路径访问内容提供者,返回查询结果集
// 参数:路径,返回字段,条件字段,条件字段值,排序
// 相当于: select 返回字段 from 路径 where 条件字段=条件字段值 order by 排序
Cursor cursor = resolver.query(uri, null, null, null, null);
while (cursor.moveToNext()) {
// 长度可变的字符串
StringBuilder sb = new StringBuilder();
// 获得联系人的id和姓名
String contactId = cursor.getString(cursor
.getColumnIndex(ContactsContract.Contacts._ID));
String name = cursor.getString(cursor
.getColumnIndex(ContactsContract.Contacts.DISPLAY_NAME));
sb.append("id=").append(contactId).append(",name=").append(name + "\n");

// 获得联系人的手机号码(一个联系人可能有多个手机号码)
Cursor phones = resolver.query(
ContactsContract.CommonDataKinds.Phone.CONTENT_URI,
null,
ContactsContract.CommonDataKinds.Phone.CONTACT_ID
+ " = " + contactId, null, null);
while (phones.moveToNext()) {
String phoneNumber = phones.getString(phones
.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER));
sb.append(",phone=").append(phoneNumber);
}
phones.close();
text += sb.toString() + "\n";
}
cursor.close();

tvContact.setText("");
tvContact.append(text);
}

}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值