内容提供者
2016-06-30 00:07
357 查看
内容提供者(相当于中间人)
应用的数据库是不允许其他应用访问的内容提供者的作用就是让别的应用访问到你的数据库
自定义内容提供者,继承ContentProvider类,重写增删改查方法,在方法中写增删改查数据库的代码,举例增方法
@Override public Uri insert(Uri uri, ContentValues values) { db.insert("person", null, values); return uri; }
在清单文件中定义内容提供者的标签,注意必须要有authorities属性,这是内容提供者的主机名,功能类似地址
<provider android:name="com.itheima.contentprovider.PersonProvider" android:authorities="com.itheima.person" android:exported="true" ></provider>
创建一个其他应用,访问自定义的内容提供者,实现对数据库的插入操作
public void click(View v){ //得到内容分解器对象 ContentResolver cr = getContentResolver(); ContentValues cv = new ContentValues(); cv.put("name", "小方"); cv.put("phone", 138856); cv.put("money", 3000); //url:内容提供者的主机名 cr.insert(Uri.parse("content://com.itheima.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);
短信数据库
只需要关注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,并且在条件里面附加deleted字段为0(deleted字段是判断在手机上面联系人是否被删除,是就为1,否则为0)如果手机的联系人被删除的话,但是在数据库中仍然存在该联系人信息,只不过是删除了content_id这个字段而已,目的是为了联系人云端能够快速找到被删除的联系人,然后备份(ios也是如此)
Cursor cursor = cr.query(Uri.parse("content://com.android.contacts/raw_contacts"), new String[]{"contact_id"},"deleted=?", new String[]{"0"}, 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应该是多少(因为contcct_id可能为空)ContentResolver cr = getContentResolver(); //先查询raw_contacts表,获取最新联系人的主键,然后主键+1,就是要插入的联系人的id Cursor cursorContactId = cr.query(Uri.parse("content://com.android.contacts/raw_contacts"), new String[]{"_id"}, null, null, null); //默认联系人id就是1 int contact_id = 1; if(cursorContactId.moveToLast()){ //拿到主键 int _id = cursorContactId.getInt(0); //主键+1,就是要插入的联系人id contact_id = ++_id; } //记得关闭游标 cursorContactId.close();
把确定的联系人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);
相关文章推荐
- Android数据持久化存储
- 《剑指offer》二叉树的镜像
- Android开发小问题记录
- android Activity 四大启动模式探究
- 软件项目风险管理(Project Risk Management)
- 广播
- 随便写给phper的docker入门手册
- Floyd最短路算法
- mysql使用出现的问题
- 安全、体验、技术至上:刷脸支付这样一路走向我们身边
- linux解压包
- PHP请求Socket接口测试实例
- Visual Studio 2010 前端开发工具/扩展/插件推荐
- 最锋利的Visual Studio Web开发工具扩展:Web Essentials使用详解
- ASP.NET MVC3 SEO优化:利用Routing特性提高站点权重
- ASP.NET MVC中使用jQuery时的浏览器缓存问题详解
- ASP.NET中URL Routing和IIS上URL Rewriting的区别
- 详解MongoDB中创建集合与删除集合的操作方法
- MongoDB中对文档的增删查改基本操作方法总结
- 详解Python操作RabbitMQ服务器消息队列的远程结果返回