項(xiàng)目需求:
Android開發(fā)中用戶登錄往會用到短信驗(yàn)證的功能,如果讓用戶先查看短信,然后再回到界面填寫驗(yàn)證碼,用戶體驗(yàn)不是很好只怎,有時就需要實(shí)現(xiàn)驗(yàn)證碼的自動填寫功能跟衅。
做法:
我以前的做法是先創(chuàng)建一個廣播接收器,接受短信變化的廣播,收到廣播時,再把驗(yàn)證碼提取出來谤饭。
那時有用戶測試反饋他的手機(jī)安裝了其他一些短信應(yīng)用或者手機(jī)本身限制了權(quán)限的情況下,這種方式可能起不了作用炫欺,即使把優(yōu)先級設(shè)高乎完,也不能保證不會被別的應(yīng)用搶先。
以前用戶卸載掉第三方軟件后品洛,就沒再追究了树姨。
現(xiàn)在發(fā)現(xiàn)可以通過監(jiān)聽短信數(shù)據(jù)庫的方式實(shí)現(xiàn)摩桶。監(jiān)聽短信數(shù)據(jù)庫主要是通過ContentObserver這個類來完成。ContentObserver主要是通過Uri來監(jiān)測特定的Databases的表帽揪,當(dāng)ContentObserver所觀察的Uri發(fā)生變化時硝清,便會觸發(fā)它。ContentObserver內(nèi)容觀察者转晰,可監(jiān)聽觀察特定Uri指向的數(shù)據(jù)庫項(xiàng)的變化芦拿,進(jìn)而進(jìn)行相應(yīng)的處理。
public class MessageContentObserver extends ContentObserver {
private Context mContext;
private Handler mHandler;
private String code;
public MessageContentObserver(Context context, Handler handler) {
super(handler);
mContext = context;
mHandler = handler;
}
/**
* 回調(diào)函數(shù), 當(dāng)監(jiān)聽的Uri發(fā)生改變時查邢,會回調(diào)該方法
* 需要注意的是當(dāng)收到短信的時候會回調(diào)兩次
* 收到短信一般來說都是執(zhí)行了兩次onchange方法.第一次一般都是raw的這個.
* 雖然收到了短信.但是短信并沒有寫入到收件箱里
*/
@Override
public void onChange(boolean selfChange, Uri uri) {
if (uri.toString().equals("content://sms/raw")) {
return;
}
Uri inboxUri = Uri.parse("content://sms/inbox");
Cursor c = mContext.getContentResolver().query(inboxUri, null, null, null, "date desc"); // 按時間順序排序短信數(shù)據(jù)庫
if (c != null) {
if (c.moveToFirst()) {
String address = c.getString(c.getColumnIndex("address"));//發(fā)送方號碼
String body = c.getString(c.getColumnIndex("body")); // 短信內(nèi)容
if (!address.equals("10086")) {
return;
}
Pattern pattern = Pattern.compile("(\\d{6})");//正則表達(dá)式匹配驗(yàn)證碼
Matcher matcher = pattern.matcher(body); if (matcher.find()) {
code = matcher.group(0);
Message msg = Message.obtain();
msg.what = MainActivity.MSG_RECEIVE_CODE;
msg.obj = code;
mHandler.sendMessage(msg);
}
}
c.close();
}
}
}
調(diào)用:
/**
* 短信驗(yàn)證碼自動填寫功能的實(shí)現(xiàn)
*/
public class MainActivity extends Activity {
public static final int MSG_RECEIVE_CODE = 1; //收到短信的驗(yàn)證碼
private EditText codeEdt; //短信驗(yàn)證碼的輸入框
private MessageContentObserver messageContentObserver; //回調(diào)接口
@SuppressLint("HandlerLeak")
Handler handler = new Handler() {
@Override
public void handleMessage(Message msg) {
if (msg.what == MSG_RECEIVE_CODE) {
//設(shè)置讀取到的內(nèi)容
}
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
codeEdt = (EditText) findViewById(R.id.smsCode);
findViewById(R.id.send_sms_btn).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
senSMSCode();
}
});
messageContentObserver = new MessageContentObserver(MainActivity.this, handler);
getContentResolver().registerContentObserver(Uri.parse("content://sms/"), true, messageContentObserver);
}
/**
* 取消注冊
*/
@Override
protected void onDestroy() {
super.onDestroy();
getContentResolver().unregisterContentObserver(messageContentObserver);
}
private void senSMSCode() {
}
}
需要在AndroidManifest.xml加上權(quán)限
<uses-permission android:name="android.permission.RECEIVE_SMS"/>
<uses-permission android:name="android.permission.READ_SMS" />
關(guān)于content://sms/inbox表蔗崎,大致包含的域有:
_id | 短消息序號 如100 thread_id | 對話的序號 如100
address | 發(fā)件人地址,手機(jī)號.如+8613811810000 person | 發(fā)件人扰藕,返回一個數(shù)字就是聯(lián)系人列表里的序號缓苛,陌生人為null
date | 日期 long型。如1256539465022 protocol | 協(xié)議 0 SMS_RPOTO, 1 MMS_PROTO
read | 是否閱讀 0未讀邓深, 1已讀
status | 狀態(tài) -1接收未桥,0 complete, 64 pending, 128 failed
type | 類型 1是接收到的,2是已發(fā)出
body | 短消息內(nèi)容
service_center | 短信服務(wù)中心號碼編號芥备。
content://sms/inbox 收件箱
content://sms/sent 已發(fā)送
content://sms/draft 草稿
content://sms/outbox 發(fā)件箱 (正在發(fā)送的信息)
content://sms/failed 發(fā)送失敗
content://sms/queued 待發(fā)送列表
項(xiàng)目地址
https://github.com/88ios/SMSContentObserver-master
項(xiàng)目主要用第三方bomb來發(fā)短信冬耿,感興趣不妨看看。
[END]