执行效果图:
预备知识:
为了监听指定的ContentProvider的数据的改变,须要通过ContentResolver向指定Uri注冊CotentObserver监听器。ContentResolver提供了例如以下方法来注冊监听器:
publicfinal void registerContentObserver(Uriuri, boolean notifyForDescendents, ContentObserver observer)
參数:uri :该监听器所监听的ContentProvider的Uri。
notifyForDescendents :为false 表示精确匹配。即仅仅匹配该Uri,为true 表示能够同一时候匹配其派生的Uri。
observer:ContentObserver派生的监听器实例。
取消注冊监听器:
public finalvoid unregisterContentObserver(ContentObserver observer)
功能:取消对给定Uri的观察
參数: observer ContentObserver的派生类实例。
ContentObserver——内容观察者,目的是观察(捕捉)特定Uri引起的数据库的变化,继而做一些对应的处理,它类似于数据库技术中的触发器(Trigger)。当ContentObserver所观察的Uri发生变化时,便会触发它。触发器分为表触发器、行触发器。对应地ContentObserver也分为“表“ContentObserver、“行”ContentObserver,当然这是与它所监听的Uri MIME Type有
关的。
ContentObserver类介绍:
接收回调的更改内容。
必须由被加入到一个ContentObservable对象实现。
构造方法:
Public Constructors |
|
ContentObserver(Handler handler) onChange() will happen on the provider Handler. |
说明:全部 ContentObserver的派生类都须要调用该构造方法
參数:handler Handler对象。能够是主线程Handler(这时候能够更新UI 了),也能够是不论什么Handler对象。
经常用法:
Public Methods |
|
boolean |
deliverSelfNotifications() Returns true if this observer is interested in notifications for changes made through the cursor the observer is registered with. |
final void |
dispatchChange(boolean selfChange) |
void |
onChange(boolean selfChange) This method is called when a change occurs to the cursor that is being observed. |
说明:
void onChange(booleanselfChange)
功能:当观察到的Uri发生变化时,回调该方法去处理。全部ContentObserver的派生类都须要重载该方法去处理逻辑。
參数:selfChange 回调后。其值一般为false,该參数意义不大(我也不懂。理解方法最重要)。
观察特定Uri的过程例如以下:
1、 创建我们特定的ContentObserver派生类。必须重载父类构造方法。必须重载onChange()方法去处理回调后的功能实现
2、 利用context.getContentResolover()获得ContentResolove对象。接着调用registerContentObserver()方法去注冊内容观察者
如://为content://sms的数据改变注冊监听器getContentResolver().registerContentObserver(Uri.parse
("content://sms"), true, new SmsObserver(new Handler()));
3、 因为ContentObserver的生命周期不同步于Activity和Service等。因此,在不须要时,须要手动的调用unregisterContentObserver()去取消注冊。
短信相关权限:
<!-- 发送消息--><uses-permission android:name="android.permission.SEND_SMS"/>
<!-- 阅读消息-->
<uses-permission android:name="android.permission.READ_SMS"/>
<!-- 写入消息-->
<uses-permission android:name="android.permission.WRITE_SMS" />
<!-- 接收消息 -->
<uses-permission android:name="android.permission.RECEIVE_SMS"/>
相关的协议:
content://sms/sent
content://sms/draft
content://sms/outbox
content://sms/failed
content://sms/queued
数据库中sms相关的字段及说明:
字段 |
说明 |
_id |
短信序号,如100 |
thread_id |
对话的序号,如100,与同一个手机号互发的短信。其序号是同样的 |
address |
发件人地址,即手机号。如+86138138000 |
person |
发件人,假设发件人在通讯录中则为详细姓名。陌生人为null |
date |
日期。long型,如1346988516。能够对日期显示格式进行设置 |
protocol |
协议0SMS_RPOTO短信,1MMS_PROTO彩信 |
read |
是否阅读0未读,1已读 |
status |
短信状态-1接收,0complete,64pending,128failed |
type |
短信类型1是接收到的,2是已发出 |
body |
短信详细内容 |
service_center |
短信服务中心号码编号。如+8613800755500 |
应用实例:
package com.jph.monitorsms; import java.text.SimpleDateFormat; import java.util.Date; import android.net.Uri; import android.os.Bundle; import android.os.Handler; import android.widget.TextView; import android.widget.Toast; import android.app.Activity; import android.database.ContentObserver; import android.database.Cursor; /** * Describe:</br> * 获取用户正在发送的短信 * 本实例通过为content://sms的数据改变注冊监听器来 * 获取手机正在发送的消息。* @author JPH * Date:2014.07.20 * */ public class MonitorSms extends Activity { TextView txtView; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); txtView=(TextView)findViewById(R.id.txtView); //为content://sms的数据改变注冊监听器 getContentResolver().registerContentObserver(Uri.parse ("content://sms"), true, new SmsObserver(new Handler())); } //一个继承自ContentObserver的监听器类 class SmsObserver extends ContentObserver{ public SmsObserver(Handler handler) { super(handler); // TODO Auto-generated constructor stub } @Override public void onChange(boolean selfChange) { // TODO Auto-generated method stub //查询发送向箱中的短信 Cursor cursor=getContentResolver().query(Uri.parse( "content://sms/outbox"), null, null, null, null); //遍历查询结果获取用户正在发送的短信 while (cursor.moveToNext()) { StringBuffer sb=new StringBuffer(); //获取短信的发送地址 sb.append("发送地址:"+cursor.getString(cursor.getColumnIndex("address"))); //获取短信的标题 sb.append("\n标题:"+cursor.getString(cursor.getColumnIndex("subject"))); //获取短信的内容 sb.append("\n内容:"+cursor.getString(cursor.getColumnIndex("body"))); //获取短信的发送时间 Date date=new Date(cursor.getLong(cursor.getColumnIndex("date"))); //格式化以秒为单位的日期 SimpleDateFormat sdf=new SimpleDateFormat("yyyy年MM月dd日 hh时mm分ss秒"); sb.append("\n时间:"+sdf.format(date)); System.out.println("查询到的正在发送的短信:"+sb.toString()); Toast.makeText(MonitorSms.this, sb.toString(), Toast.LENGTH_LONG).show(); txtView.setText(sb.toString()); } super.onChange(selfChange); } } }