Android短信验证码自动填写功能的实现

时间:2022-11-03 20:07:32

前言:

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
[html] view plain copy
  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  2.     xmlns:tools="http://schemas.android.com/tools"  
  3.     android:layout_width="match_parent"  
  4.     android:layout_height="match_parent"  
  5.     android:paddingBottom="@dimen/activity_vertical_margin"  
  6.     android:paddingLeft="@dimen/activity_horizontal_margin"  
  7.     android:paddingRight="@dimen/activity_horizontal_margin"  
  8.     android:paddingTop="@dimen/activity_vertical_margin"  
  9.     tools:context=".MainActivity">  
  10.   
  11.     <EditText  
  12.         android:id="@+id/et_validateCode"  
  13.         android:layout_width="wrap_content"  
  14.         android:layout_height="wrap_content"  
  15.         android:layout_alignParentTop="true"  
  16.         android:layout_centerHorizontal="true"  
  17.         android:ems="10" />  
  18. </RelativeLayout>  

MainActivity.java[java] view plain copy
  1. package smsdemo.com.smsdemo;  
  2.   
  3. import android.app.Activity;  
  4. import android.net.Uri;  
  5. import android.os.Bundle;  
  6. import android.os.Handler;  
  7. import android.os.Message;  
  8. import android.widget.EditText;  
  9.   
  10. /** 
  11.  * 短信验证码自动填写功能的实现 
  12.  * 
  13.  * Created by huangminzheng on 16/3/15. 
  14.  */  
  15. public class MainActivity extends Activity {  
  16.   
  17.     public static final int MSG_RECEIVED_CODE = 1;  
  18.     private EditText metValidateCode = null;  
  19.     private SmsObserver mObserver;  
  20.   
  21.     @Override  
  22.     protected void onCreate(Bundle savedInstanceState) {  
  23.         super.onCreate(savedInstanceState);  
  24.         setContentView(R.layout.activity_main);  
  25.   
  26.         metValidateCode = (EditText) findViewById(R.id.et_validateCode);  
  27.   
  28.         mObserver = new SmsObserver(MainActivity.this, mHandler);  
  29.         Uri uri = Uri.parse("content://sms");  
  30.         //注册短信的监听  
  31.         getContentResolver().registerContentObserver(uri, true, mObserver);  
  32.     }  
  33.   
  34.     @Override  
  35.     protected void onPause() {  
  36.         super.onPause();  
  37.         //解除注册短信的监听  
  38.         getContentResolver().unregisterContentObserver(mObserver);  
  39.     }  
  40.   
  41.     private Handler mHandler = new Handler() {  
  42.         @Override  
  43.         public void handleMessage(Message msg) {  
  44.             if (msg.what == MSG_RECEIVED_CODE) {  
  45.                 String code = (String) msg.obj;  
  46.                 metValidateCode.setText(code);  
  47.             }  
  48.         }  
  49.     };  
  50.   
  51. }  

SmsObserver.java
[java] view plain copy
  1. package smsdemo.com.smsdemo;  
  2.   
  3. import android.content.Context;  
  4. import android.database.ContentObserver;  
  5. import android.database.Cursor;  
  6. import android.net.Uri;  
  7. import android.os.Handler;  
  8. import android.util.Log;  
  9. import java.util.regex.Matcher;  
  10. import java.util.regex.Pattern;  
  11.   
  12. /** 
  13.  * Created by huangminzheng on 16/3/15. 
  14.  * 
  15.  * 观察者对象 
  16.  */  
  17. public class SmsObserver extends ContentObserver{  
  18.   
  19.     private Context mContext;  
  20.     private Handler mHandler;  
  21.   
  22.     public SmsObserver(Context context, Handler handler) {  
  23.         super(handler);  
  24.         mContext = context;  
  25.         mHandler = handler;  
  26.     }  
  27.   
  28.     @Override  
  29.     public void onChange(boolean selfChange, Uri uri) {  
  30.         super.onChange(selfChange, uri);  
  31.   
  32.         Log.d("main""SMS has changed!");  
  33.         Log.d("main", uri.toString());  
  34.         // 短信内容变化时,第一次调用该方法时短信内容并没有写入到数据库中,return  
  35.         if (uri.toString().equals("content://sms/raw")) {  
  36.             return;  
  37.         }  
  38.         getValidateCode();//获取短信验证码  
  39.   
  40.     }  
  41.   
  42.     /** 
  43.      * 获取短信验证码 
  44.      */  
  45.     private void getValidateCode() {  
  46.         String code = "";  
  47.         Uri inboxUri = Uri.parse("content://sms/inbox");  
  48.         Cursor c = mContext.getContentResolver().query(inboxUri, nullnullnull"date desc");//  
  49.         if (c != null) {  
  50.             if (c.moveToFirst()) {  
  51.                 String address = c.getString(c.getColumnIndex("address"));  
  52.                 String body = c.getString(c.getColumnIndex("body"));  
  53.   
  54.                 //13162364720为发件人的手机号码  
  55.                 if (!address.equals("13162364720")) {  
  56.                     return;  
  57.                 }  
  58.                 Log.d("main""发件人为:" + address + " ," + "短信内容为:" + body);  
  59.   
  60.                 Pattern pattern = Pattern.compile("(\\d{6})");  
  61.                 Matcher matcher = pattern.matcher(body);  
  62.   
  63.                 if (matcher.find()) {  
  64.                     code = matcher.group(0);  
  65.                     Log.d("main""验证码为: " + code);  
  66.                     mHandler.obtainMessage(MainActivity.MSG_RECEIVED_CODE, code).sendToTarget();  
  67.                 }  
  68.   
  69.             }  
  70.             c.close();  
  71.         }  
  72.     }  
  73. }  

短信的Uri共有一下几种:
[plain] view plain copy
  1. content://sms/inbox     收件箱           
  2. content://sms/sent      已发送   
  3. content://sms/draft     草稿             
  4. content://sms/outbox    发件箱  (正在发送的信息)  
  5. content://sms/failed    发送失败       
  6. content://sms/queued    待发送列表  (比如开启飞行模式后,该短信就在待发送列表里)  

当然不要忘记添加读取短信的权限:
[html] view plain copy
  1. <uses-permission android:name="android.permission.READ_SMS" />  

源码:点击下载