之前第三章理论知识写到过数据库。数据库是在程序内部自己访问自己。而内容提供器是访问别的程序数据的,即跨程序共享数据。对访问的数据也无非就是CRUD。
内容提供者
- 应用的数据库是不允许其他应用访问的
- 内容提供者的作用就是让别的应用访问到你的数据库
- 写自定义内容提供者的代码就是在被访问程序与主访问程序之间交替写代码。
-
自定义内容提供者,继承ContentProvider类,重写增删改查方法,在方法中写增删改查数据库的代码,举例增方法。自定义继承使用ContentProvider
@Override public Uri insert(Uri uri, ContentValues values) { db.insert("person", null, values); return uri; } -
在清单文件中定义内容提供者的标签,注意必须要有authorities属性,这是内容提供者的主机名,功能类似地址
<provider android:name="com.it.contentprovider.PersonProvider" android:authorities="com.it.person" android:exported="true" ></provider> -
创建一个其他应用,访问自定义的内容提供者,实现对数据库的插入操作
public void click(View v){ //得到内容分解器对象 ContentResolver cr = getContentResolver();//访问使用ContentResolver ContentValues cv = new ContentValues(); cv.put("name", "小方"); cv.put("phone", 138856); cv.put("money", 3000); //url:内容提供者的主机名,这里的uri与上边配置文件中的一样 cr.insert(Uri.parse("content://com.it.person"), cv); }
UriMatcher
- 用于判断一条uri跟指定的多条uri中的哪条匹配
-
添加匹配规则
//指定多条uri um.addURI("com.itheima.person", "person", PERSON_CODE); um.addURI("com.itheima.person", "company", COMPANY_CODE); //#号可以代表任意数字 um.addURI("com.itheima.person", "person/#", QUERY_ONE_PERSON_CODE); -
通过Uri匹配器可以实现操作不同的表
@Override public Uri insert(Uri uri, ContentValues values) { if(um.match(uri) == PERSON_CODE){ db.insert("person", null, values); } else if(um.match(uri) == COMPANY_CODE){ db.insert("company", null, values); } else{ throw new IllegalArgumentException("不匹配"); } return uri; } -
如果路径中带有数字,把路径末尾的数字提取出来的api
int id = (int) ContentUris.parseId(uri);//返回的long类型,强制转换int
短信数据库
- 只需要关注sms表
-
只需要关注4个字段
- body:短信内容
- address:短信的发件人或收件人号码(跟你聊天那哥们的号码)
- date:短信时间
- type:1为收到,2为发送
-
读取系统短信,首先查询源码获得短信数据库内容提供者的主机名和路径,然后
ContentResolver cr = getContentResolver(); Cursor c = cr.query(Uri.parse("content://sms"), new String[]{"body", "date", "address", "type"}, null, null, null); while(c.moveToNext()){ String body = c.getString(0); String date = c.getString(1); String address = c.getString(2); String type = c.getString(3); System.out.println(body+";" + date + ";" + address + ";" + type); } -
插入系统短信
ContentResolver cr = getContentResolver(); ContentValues cv = new ContentValues(); cv.put("body", "您尾号为XXXX的招行储蓄卡收到转账1,000,000人民币"); cv.put("address", 95555); cv.put("type", 1); cv.put("date", System.currentTimeMillis()); cr.insert(Uri.parse("content://sms"), cv); - 插入查询系统短信需要注册权限
联系人数据库
-
raw_contacts表:
- contact_id:联系人id
-
data表:联系人的具体信息,一个信息占一行
- data1:信息的具体内容
- raw_contact_id:联系人id,描述信息属于哪个联系人
- mimetype_id:描述信息是属于什么类型
- mimetypes表:通过mimetype_id到该表查看具体类型
读取联系人
-
先查询raw_contacts表拿到联系人id
Cursor cursor = cr.query(Uri.parse("content://com.android.contacts/raw_contacts"), new String[]{"contact_id"}, null, null, null); -
然后拿着联系人id去data表查询属于该联系人的信息
Cursor c = cr.query(Uri.parse("content://com.android.contacts/data"), new String[]{"data1", "mimetype"}, "raw_contact_id = ", new String[]{contactId}, null); -
得到data1字段的值,就是联系人的信息,通过mimetype判断是什么类型的信息
while(c.moveToNext()){ String data1 = c.getString(0); String mimetype = c.getString(1); if("vnd.android.cursor.item/email_v2".equals(mimetype)){ contact.setEmail(data1); } else if("vnd.android.cursor.item/name".equals(mimetype)){ contact.setName(data1); } else if("vnd.android.cursor.item/phone_v2".equals(mimetype)){ contact.setPhone(data1); } }
插入联系人
- 先查询raw_contacts表,确定新的联系人的id应该是多少
-
把确定的联系人id插入raw_contacts表
cv.put("contact_id", _id); cr.insert(Uri.parse("content://com.android.contacts/raw_contacts"), cv); -
在data表插入数据
-
插3个字段:data1、mimetype、raw_contact_id
cv = new ContentValues(); cv.put("data1", "赵六"); cv.put("mimetype", "vnd.android.cursor.item/name"); cv.put("raw_contact_id", _id); cr.insert(Uri.parse("content://com.android.contacts/data"), cv); cv = new ContentValues(); cv.put("data1", "1596874"); cv.put("mimetype", "vnd.android.cursor.item/phone_v2"); cv.put("raw_contact_id", _id); cr.insert(Uri.parse("content://com.android.contacts/data"), cv);
-
内容观察者
-
当数据库数据改变时,内容提供者会发出通知,在内容提供者的uri上注册一个内容观察者,就可以收到数据改变的通知
cr.registerContentObserver(Uri.parse("content://sms"), true, new MyObserver(new Handler())); class MyObserver extends ContentObserver{ public MyObserver(Handler handler) { super(handler); // TODO Auto-generated constructor stub } //内容观察者收到数据库发生改变的通知时,会调用此方法 @Override public void onChange(boolean selfChange) { } } -
在内容提供者中发通知的代码
ContentResolver cr = getContext().getContentResolver(); //发出通知,所有注册在这个uri上的内容观察者都可以收到通知 cr.notifyChange(uri, null);
- 顶
- 1
- 踩
- 0
下一篇:RxJava操作符(02-创建操作)
-
从0快速搭建一个实用的MVVM框架(超详细)
这篇文章主要介绍了从0搭建一个实用的MVVM框架,结合Jetpack,构建快速开发的MVVM框架,支持快速生成ListActivity、ListFragment,主要是基于MVVM进行快速开发上手即用
-
Flutter绘图组件之CustomPaint使用详细介绍
CustomPaint是Flutter中用于自由绘制的一个widget,它与android原生的绘制规则基本一致,以当前Canves(画布)的左上角为原点进行绘制。本文将详细讲解CustomPaint
-
Android开发使用WebView打造webapp示例代码
这篇文章主要介绍了Android开发使用WebView打造webapp的关键示例代码,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步
-
Android开发之自定义加载动画详解
这篇文章主要介绍了Android开发的自定义加载动画,效果为一个连续的动画,就是这个大圆不停地吞下小圆,文中示例代码讲解详细,感兴趣的可以了解一下
-
开源直播系统源码功能一览表(含ios+android+服务端+后台)
 在购买源码之时切记不要贪便宜,避免偷鸡不成蚀把米,源码都是由程序员编写而出,花费了时间与心血,所以网上所谓的泄露版、破解版、免费提供都不可相信,适合自己的直播源码才是最好的。
- 新闻发布系统
- 个人网站源码
- 个人主页源码
- 帝国模板
- 企业网站管理
- .net源码
- 办公管理系统
- php 源码
- 网站源码免费下载
- 广告联盟源码
- 网络编程技术
- web源码
- discuz下载
- 门户网站源码
- 蜘蛛bt
- 登录界面模板
- 模板之家
- 下载网站源码
- 导航条代码
- 网站右下角广告
- 源代码下载
- jquery api
- PHP源码
- 简单网页模板
- asp.net源码
- 整站程序
- 免费网页模板
- 源码站
- 源码天空
- php 框架
- 网页 模板
- 安卓源码下载
- php网站源码
- 电脑报电子版
- 源码分享
- 免费网站模板下载
- asp cms
- 网站下载
- 安卓源码
- 网站模版下载
- 源码之家
- 导航网站模板
- 素材解析平台
- 网页作业
- 模版代码
- 源码
- 网站源码出售
- 精品源码
- php论坛
- 网站首页模板