Android开发之自动填充短信验证码

作者: YungFan | 来源:发表于2016-04-10 07:11 被阅读2925次

    笔者发现在很多应用中,都有自动获取验证码的功能:点击获取验证码按钮,收到短信,当前应用不需要退出程序就可以获取到短信中的验证码,并自动填充。觉得这种用户体验很赞,无须用户在短信和App之间来回切换,这个功能是如何实现的呢?其实很简单,就是用到了Android中的一个叫ContentObserver的东西,下面手动简单实现一个~~~

    </br>

    一、开发ContentObserver

    主要是用来监听收件箱的内容,一旦有新消息过来,就去监听是否是我想要的那个号码发过来的短信,如果是,就直接用正则表达式截取

    /**
     * 短信验证码截取
     * 
     * @author yangfan
     * 
     */
    public class SMSContentObserver extends ContentObserver
    {
    
        private Context mContext; // 上下文
        private Handler mHandler; // 更新UI线程
        private String code; // 验证码
    
        public SMSContentObserver(Context context, Handler handler)
        {
            super(handler);
            mContext = context;
            mHandler = handler;
        }
    
        /**
         * 回调函数, 当所监听的Uri发生改变时,就会回调此方法
         * 
         * 注意当收到短信的时候会回调两次
         * 
         * @param selfChange
         *            此值意义不大 一般情况下该回调值false
         */
        @Override
        public void onChange(boolean selfChange, Uri uri)
        {
    
            Log.e("XXXXXXXXXXXXXXXX", uri.toString());
    
            // 第一次回调 不是我们想要的 直接返回
            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");
            if (c != null)
            {
                if (c.moveToFirst())
                {
    
                    // 获取手机号
                    String address = c.getString(c.getColumnIndex("address"));
                    // 获取短信内容
                    String body = c.getString(c.getColumnIndex("body"));
                    // 判断手机号是否为目标号码
                    if (!address.equals("你的目标号码"))
                    {
                        return;
                    }
    
                    // 正则表达式截取短信中的6位验证码
                    Pattern pattern = Pattern.compile("(\\d{6})");
                    Matcher matcher = pattern.matcher(body);
    
                    // 如果找到通过Handler发送给主线程
                    if (matcher.find())
                    {
                        code = matcher.group(0);
                        mHandler.obtainMessage(1, code).sendToTarget();
                    }
                }
    
            }
            c.close();
    
        }
    
    }
    
    
    二、布局和Activity
    <LinearLayout 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:padding="50dp" >
    
        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="您的验证码为:" />
    
        <EditText
            android:id="@+id/smsCode"
            android:focusable="false"
            android:focusableInTouchMode="false"
            android:layout_width="match_parent"
            android:layout_height="wrap_content" />
    
    </LinearLayout>
    
    public class MainActivity extends ActionBarActivity
    {
        private EditText code;
    
        @SuppressLint("HandlerLeak")
        Handler handler = new Handler()
        {
            @Override
            public void handleMessage(Message msg)
            {
                if (msg.what == 1)
                {
                    code.setText(msg.obj.toString());
                }
            }
        };
    
        @Override
        protected void onCreate(Bundle savedInstanceState)
        {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
    
            code = (EditText) findViewById(R.id.smsCode);
    
            SMSContentObserver smsContentObserver = new SMSContentObserver(
                    MainActivity.this, handler);
    
            MainActivity.this.getContentResolver().registerContentObserver(
                    Uri.parse("content://sms/"), true, smsContentObserver);
        }
    }
    
    
    三、不要忘记加权限
      <uses-permission android:name="android.permission.READ_SMS" />
    
    
    四、运行测试

    本人是拿两部手机来测试的 ~~~

    自动获取短信验证码.png

    代码地址:https://github.com/yungfan/SMSSMSContentObserver

    相关文章

      网友评论

      • senmuZ:请问这个在华为手机上获取不到验证码类的短信,有什么方法可以解决吗?
        YungFan:@senmuZ 底层改了太多了的系统都没办法 很多功能在原生的Android系统都没问题
        senmuZ:@YungFan 感觉是被华为系统给拦截掉了
        小米系统有个打开获取通知类短信的权限,但是华为的系统好像没有
        YungFan:@senmuZ 检查系统权限 应该是华为系统版本的问题
      • cd0328bceee3:你好,我是个新人,刚接触这个,很好奇,请告诉我下载了你的代码后怎么办,怎么用
        YungFan:@每日一格 6.0以后需要动态权限判断
        37f957305ae1:安卓6.0不能自动填充,而且崩溃。。。 另一台手机5.0的不崩溃,但是不能自动填充
        YungFan:@原谅我不懂 当时用ADT写的 你可以直接用ADT导入或者用AS导入ADT项目
      • 7571e3d5b51b:我咋填充不进去呢
        YungFan:@峰仔_a694 不应该,报错了没
      • 9be664ca7f5d:这个方法在android6.0以下可以使用 但是android6.0的时候就会崩溃 有没有什么好的结局办法?
        YungFan:@璋霏 你就对6.0单独判断一下 如果不行 估计就被废弃了
        9be664ca7f5d:@YungFan 这个方法只能保证在android6.0不会蹦 但是好像还是不能自动填写验证码 :flushed:
        YungFan:@璋霏 加个权限申请
      • 陈树鸿:我觉得用户会很反感需要开启读取短信权限的应用
        YungFan:@陈树鸿 并没有做流氓行为的事
      • 傅之橙:6位连续数字正则表达式"\\d{6}"
        YungFan:@傅之橙 是的 估计是我贴的时候出问题了
      • Razerdp:mark
      • 慕空休竹:有用
        YungFan:@FTF 谢谢
      • 54577a59080c:好像获取不到啊
        YungFan:@Sing_Ls 手机号改了?
        54577a59080c: @YungFan 你的源码也测试,没通过
        YungFan:@Sing_Ls 不可能 请下载我的源码
      • 04da40e51179:t不错
      • a521wangzhaof:极客学院的那个吧,写的挺好的
        YungFan:@a521wangzhaof :blush:
      • cff43f8d1cab:对于为什么注册的时候使用的URI是"content://sms/",而直接是"content://sms/inbox",考虑后来者可能有同样的疑问。在这里做个笔记:对短消息的观察Uri,通过测试发现只能监听此Uri “content://sms/” ,而不能监听其他的Uri,比如"content://sms/inbox"等。 :smile:
        YungFan:@ListenAlone :+1:

      本文标题:Android开发之自动填充短信验证码

      本文链接:https://www.haomeiwen.com/subject/bvvmlttx.html