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短信验证码自动填写
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。