为了监听指定的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有
关的。
接收回调的更改内容。必须由被添加到一个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()去取消注册。
字段 |
说明 |
_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); } } }