Android开发--详解ContentProvider/Cursor的使用
ContentProvider是Android四大组件之一,所以如果是自己实现ContentProvider,需要在AndroidManifest.xml文件中进行声明,幸运的是,我们很少需要自己定义实现它,一般我们的开发只需要用到系统自己提供的ContentProvider,使用起来非常方便。
首先来让我们认识ContentProvider:
基本概念:
1.ContentProvider提供为存储和获取数据提供了统一的接口
2.使用ContentProvider可以在不同的应用程序之间共享数据
3.Android为常见的一些数据提供了ContentProvider
下面我开始介绍实现ContentProvider的过程:
1.定义一个CONTENT_URI常量
2.定义一个雷,继承自ContentProvider
3.实现query,insert,update,delete,getType和onCreate方法
4.在AndroidManifest.xml文件中进行声明
ContentProvider提供了一系列的函数,当自己实现ContentProvider需要实现这些方法:
1.query()
2.insert()
3.update()
4.delete()
5.getType()
6.onCreate()
接下来详细介绍ContentProvider:
*一.初识Content Provider
* 1.简介
* 1.1 Content Provider实现了一组通用的方法来提供数据的增删改查的功能
* 1.2 客户端通常不会直接使用这些方法,大多数都是通过ContentResolver对象实现对Content Provider的操作
* 1.3 开发一般会通过调用方法获得ContentProvider对象
* 例如:ContentResolver cr=getContentResolver
* 使用ContentResolver提供的方法可以获得ContentProvider中任何感兴趣的对象
* 2.数据模型
* 2.1 ContentProvider使用基于数据库模型的简单表格来提供其中的数据,其中每一行都代表一个记录,
* 每一列代表特定类型和含义的数据,其中每条记录包含一个数值型的_ID字段,用于在表格中唯一标识该记录
* 注意:ID字段前还包含了一个下划线
* 2.2 查询返回一个Cursor对象,它能遍历各行各列来读取每个字段的值。对于各个类型的数据,Cursor对象都提供了
* 专用的方法,因此,为了读取字段的数据,开发人员必须要知道当前字段包含的数据类型
* 注意:Cursor的使用方法:通过查询,它会指向第一个记录之前,所以需要开发者moveToNext,且查询字段的动作是
* 先指向某一条记录,即指向某一行,再从某一列或多列中取得数据,且需要知道数据类型(列数据),例如:
* int id=cursor.getInt(idIndex);
* String name=cursor.getString(displayNameIndex);
* 3.URI的用法
* 3.1 每个Content Provider提供公共的URI(使用Uri类包装)来唯一标识其数据集。管理多个数据集(多个表格)的
* Content Provider为每个都提供了单独的URI,且URI的标识有固定的格式
* 3.2 UIR常量用于所有与Content Provider的交互中,每个ContentResolver方法使用URI作为第一个参数,它标识
* ContentResolver应该使用哪个provider以及其中的哪个表格
* 二.详解Content Provider
* 1.系统预定义的Content Provider
* 1.1 Android系统为常用数据类型提供了很多预定义的Content Provider(声音,视频,图片,联系人等),它们大都位于
* android.provider包中
* 1.2 Android系统提供的常见Content Provider说明如下:
* Browser:读取或修改书签,浏览历史或网络搜索
* CallLog:查看或更新通话历史
* Contacts:获取,修改或保存联系人信息
* LiveFolders:由Content Provider提供内容的特定文件夹
* MediaStore:访问声音,视频和图片
* Setting:查看和获取蓝牙设置,铃声和其他设置偏好
* SearchRecentSuggestions:该类能为应用程序创建简单的查询建议提供者
* SyncStateContract:用于使用数据数组账号关联数据的ContentProvider约束
* UserDictionary:在可预测文本输入时,提供用户定义的单词给输入法使用
* 2. 查询数据
* 2.1 在Content Provider中查询数据,开发人员需要知道一下信息:
* 标识该Content Provider的URI
* 需要查询的数据字段名称
* 字段中数据的类型
* 如果需要查询特定记录,那么还需要知道该记录的ID值
* 2.2 ContentResolver.query()或Activity.managedQuery()方法都可以完成查询功能,这两个方法
* 使用相同的参数,并且都返回Cursor对象。其区别在于managedQuery()方法让Activity来管理Cursor的
* 声明周期,而query()方法需要程序员自己管理。
* 2.3 query()方法介绍
* 方法的声明如下:
* public final Cursor query(Uri uri, String[] projection, String selection,
* String[] selectionArgs, String sortOrder)
* uri:用于查询的Content Provider的URI值
* projection:由需要查询的列名组成的数组,如果为null则表示查询全部列
* selection:类似SQL中的WHERE子句,用于增加条件来完成数据过滤
* selectionArgs:用于替换selection中可以使用?表示的变量值
* sortOrder:用于实现排序功能
* 返回值:Cursor对象,它位于第一条记录之前,或者为null
* 2.4 利用游标工具进行查询时,注意是两次查询,首先到达的那条记录的行
* 第一次查询找到需要数据的那一列,第二次查询才是真正从那一列中获得数据
* 三.总结
* 3.1 Content Provider是Android四大基本组件之一,它主要用于在不同的应用程序之间共享数据
* 3.2 Content Provider使用基于数据库模型的简单表格来提供其中的数据,这里每行代表一条记录,每列
* 代表特定类型和含义的数据
下面我用一个例子来说明ContentProvider的使用:
我们在开发的过程中经常用到联系人的信息,所以,取得联系人的信息并操纵就显得很重要,下面的例子就是对联系人的信息进行相关的操作。
下面的截图是程序的实现:
实现的源代码如下:
public class Content_Provider_Activity extends Activity {private ListView listView;private ListView listView2;private Button button;/* * 1.静态常量的定义 * 1.1 各个参数用于选择的目标,如果在数组中没有某个参数,则游标不会定义(即找到那个参数) * 1.2 从定义可以看出,联系人的所有信息不是定义在一张表之中 * 1.3 为了操作的方便和可行,分开定义查找的目标 */public static final String[] COLUMNS={Contacts._ID,Contacts.DISPLAY_NAME};private static final String[] COLUMNS2={Contacts.DISPLAY_NAME,Contacts._ID};private static final String[] COLUMNS3={Phone.NUMBER};@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_content__provider_);listView=(ListView)findViewById(R.id.listview);listView2=(ListView)findViewById(R.id.listview2);button=(Button)findViewById(R.id.button);button.setText("点击获取联系人姓名和手机号码");button.setOnClickListener(new MyButton());@SuppressWarnings("deprecation")Cursor cursor=managedQuery(Contacts.CONTENT_URI, COLUMNS, null, null, null);//获取ID所对应的索引值---列索引int idIndex=cursor.getColumnIndex(COLUMNS[0]);//获取NAME所对应的索引值---列索引int displayNameIndex=cursor.getColumnIndex(COLUMNS[1]);List<String> items=new ArrayList<String>();for(cursor.moveToFirst();!cursor.isAfterLast();cursor.moveToNext()){int id=cursor.getInt(idIndex);String name=cursor.getString(displayNameIndex);items.add("id="+id+"\t 姓名= "+name);}ArrayAdapter<String> adapter=new ArrayAdapter<String>(Content_Provider_Activity.this,R.layout.list_item,items);listView.setAdapter(adapter);}public class MyButton implements OnClickListener{@Overridepublic void onClick(View arg0) {// TODO Auto-generated method stubswitch (arg0.getId()) {case R.id.button:getNameAndPhone();break;default:break;}}}public void getNameAndPhone(){@SuppressWarnings("deprecation")Cursor cursor=managedQuery(Contacts.CONTENT_URI, COLUMNS2, null, null, null);int displayNameIndex=cursor.getColumnIndex(COLUMNS2[0]);List<String> items=new ArrayList<String>();for(cursor.moveToFirst();!cursor.isAfterLast();cursor.moveToNext()){String name=cursor.getString(displayNameIndex);int id=cursor.getInt(cursor.getColumnIndex(COLUMNS2[1]));@SuppressWarnings("deprecation")Cursor phone=managedQuery(Phone.CONTENT_URI, null, Phone.CONTACT_ID+" = "+id, null, null);while(phone.moveToNext()){String phoneNumber=phone.getString(phone.getColumnIndex(COLUMNS3[0]));items.add("姓名:"+name+"\t 手机:"+phoneNumber);}}ArrayAdapter<String> adapter=new ArrayAdapter<String>(Content_Provider_Activity.this, R.layout.list_item ,items);listView2.setAdapter(adapter);}}