編輯:關於Android編程
android應用經常會涉及到注冊登錄功能,而許多的注冊登錄或修改密碼功能常常需要輸入短信驗證碼,通常,用戶收到短信需要最小化應用去查看短信再填入驗證碼,必然比較麻煩,因此有必要能夠自動獲得下發的短信驗證碼,方便了用戶的操作,用戶體驗更好。
原理講解:
主要就是實時獲取短信信息。涉及到ContentObserver類的使用。使用ContentProvider來監聽短信數據庫的變化,在自定義的ContentObserver當中實現onChange的方法進行監聽特定手機號的短信,然後進行信息截取在填充到需要填充的位置。
ContentObserver即為內容監聽者,當我們發送一條短信到手機上時,手機會自動調用ContentObserver中的指定方法用來通知短信發生了變化,接著我們讀取短信中的內容,將驗證碼提取出來自動填入到輸入框中,這樣就完成了自動填寫功能。ContentObserver類主要監聽短信內容的變化,這裡涉及到android常用的一種設計模式即觀察者模式。
ContentObserver講解-觀察者模式:
觀察者模式(有時又被稱為發布(publish )-訂閱(Subscribe)模式、模型-視圖(View)模式、源-收聽者(Listener)模式或從屬者模式)是軟件設計模式的一種。在此種模式中,一個目標物件管理所有相依於它的觀察者物件,並且在它本身的狀態改變時主動發出通知。這通常透過呼叫各觀察者所提供的方法來實現。此種模式通常被用來實現事件處理系統。
觀察者模式(Observer)完美的將觀察者和被觀察的對象分離開。觀察者模式在模塊之間劃定了清晰的界限,提高了應用程序的可維護性和重用性。
觀察者設計模式定義了對象間的一種一對多的依賴關系,以便一個對象的狀態發生變化時,所有依賴於它的對象都得到通知並自動刷新。
ContentObserver——內容觀察者,目的是觀察(捕捉)特定Uri引起的數據庫的變化,繼而做一些相應的處理,它類似於數據庫技術中的觸發器(Trigger),當ContentObserver所觀察的Uri發生變化時,便會觸發它。
•觀察者(即我們的應用):Observer)將自己注冊到被觀察對象(Subject)中,被觀察對象將觀察者存放在一個容器(Container)裡。
•被觀察(即系統的短信應用):被觀察對象發生了某種變化(如圖中的SomeChange),從容器中得到所有注冊過的觀察者,將變化通知觀察者。
•撤銷觀察:觀察者告訴被觀察者要撤銷觀察,被觀察者從容器中將觀察者去除。
具體到我們的項目中,也就是說,當應用剛開始運行的時候,會向我們手機系統的短信應用注冊一個觀察者,當短信發生變化的時候,短信應用會通知所注冊的觀察者發生了變化,我們的觀察者收到這樣的通知時,就會根據代碼執行相應的操作,從而實現相關自動填寫驗證碼的功能。當我們完成所需要的功能時,我們要撤銷觀察,解除注冊,被觀察者從容器中將觀察者去除。觀察者被撤銷後不再收到短信的內容變化通知。
觀察特定Uri的步驟如下:
1.創建我們特定的 ContentObserver 派生類,必須重載父類構造方法,必須重載 onChange() 方法去處理回調後的功能實現。
2.利用 context.getContentResolover() 獲得 ContentResolove 對象,接著調用 registerContentObserver() 方法去注冊內容觀察者。
3.由於 ContentObserver 的生命周期不同步於 Activity 和 Service 等,因此,在不需要時,需要手動的調用 unregisterContentObserver() 去取消注冊。
activity_main.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingBottom="@dimen/activity_vertical_margin" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" tools:context=".MainActivity"> <EditText android:id="@+id/et_validateCode" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentTop="true" android:layout_centerHorizontal="true" android:ems="10" /> </RelativeLayout>
MainActivity.java
package smsdemo.com.smsdemo; import android.app.Activity; import android.net.Uri; import android.os.Bundle; import android.os.Handler; import android.os.Message; import android.widget.EditText; /** * 短信驗證碼自動填寫功能的實現 * * Created by huangminzheng on 16/3/15. */ public class MainActivity extends Activity { public static final int MSG_RECEIVED_CODE = 1; private EditText metValidateCode = null; private SmsObserver mObserver; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); metValidateCode = (EditText) findViewById(R.id.et_validateCode); mObserver = new SmsObserver(MainActivity.this, mHandler); Uri uri = Uri.parse("content://sms"); //注冊短信的監聽 getContentResolver().registerContentObserver(uri, true, mObserver); } @Override protected void onPause() { super.onPause(); //解除注冊短信的監聽 getContentResolver().unregisterContentObserver(mObserver); } private Handler mHandler = new Handler() { @Override public void handleMessage(Message msg) { if (msg.what == MSG_RECEIVED_CODE) { String code = (String) msg.obj; metValidateCode.setText(code); } } }; }
SmsObserver.java
package smsdemo.com.smsdemo; import android.content.Context; import android.database.ContentObserver; import android.database.Cursor; import android.net.Uri; import android.os.Handler; import android.util.Log; import java.util.regex.Matcher; import java.util.regex.Pattern; /** * Created by huangminzheng on 16/3/15. * * 觀察者對象 */ public class SmsObserver extends ContentObserver{ private Context mContext; private Handler mHandler; public SmsObserver(Context context, Handler handler) { super(handler); mContext = context; mHandler = handler; } @Override public void onChange(boolean selfChange, Uri uri) { super.onChange(selfChange, uri); Log.d("main", "SMS has changed!"); Log.d("main", uri.toString()); // 短信內容變化時,第一次調用該方法時短信內容並沒有寫入到數據庫中,return if (uri.toString().equals("content://sms/raw")) { return; } getValidateCode();//獲取短信驗證碼 } /** * 獲取短信驗證碼 */ private void getValidateCode() { String code = ""; Uri inboxUri = Uri.parse("content://sms/inbox"); Cursor c = mContext.getContentResolver().query(inboxUri, null, null, null, "date desc");// if (c != null) { if (c.moveToFirst()) { String address = c.getString(c.getColumnIndex("address")); String body = c.getString(c.getColumnIndex("body")); //13162364720為發件人的手機號碼 if (!address.equals("13162364720")) { return; } Log.d("main", "發件人為:" + address + " ," + "短信內容為:" + body); Pattern pattern = Pattern.compile("(\\d{6})"); Matcher matcher = pattern.matcher(body); if (matcher.find()) { code = matcher.group(0); Log.d("main", "驗證碼為: " + code); mHandler.obtainMessage(MainActivity.MSG_RECEIVED_CODE, code).sendToTarget(); } } c.close(); } } }
短信的Uri共有一下幾種:
content://sms/inbox 收件箱
content://sms/sent 已發送
content://sms/draft 草稿
content://sms/outbox 發件箱 (正在發送的信息)
content://sms/failed 發送失敗
content://sms/queued 待發送列表 (比如開啟飛行模式後,該短信就在待發送列表裡)
當然不要忘記添加讀取短信的權限:
<uses-permission android:name="android.permission.READ_SMS" />
源碼下載:Android短信驗證碼自動填寫
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持本站。
本文實例講述了Android基於反射技術實現的加減乘除運算。分享給大家供大家參考,具體如下:JAVA反射機制定義:JAVA反射機制是在運行狀態中,對於任意一個類,都能夠知
一、Android 提供了三種方式: android語音識別方法一:使用intent調用語音識別程序 1. 說明 以下例程功能為:在應用程序中使用inte
通過SpannableStringBuilder來實現,它就像html裡邊的元素改變指定文字的文字顏色或背景色public class MainActivity exte
相關閱讀:ReactiveCocoa代碼實踐之-更多思考ReactiveCocoa代碼實踐之-RAC網絡請求重構這一節是自己對網絡層的一些重構,本節是自己一些代碼小實