Android实战简易课程-第三十六枪(监听短信-实现短信验证码自动填入)

Android实战简易教程-第三十六枪(监听短信-实现短信验证码自动填入)

一般用户喜欢用手机号作为用户名注册APP账号,这时一般都是通过手机验证码的方式进行验证,下面我们就研究一个非常实用的方法,通过监听短信-实现短信验证码的自动填入,提高用户体验。
首先我们看一下如何监听手机短信。

一、获取短信全部内容

1.新建一个SMSBroadcastReceiver:

package com.example.messagecut;
import java.text.SimpleDateFormat;
import java.util.Date;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.telephony.SmsMessage;
/**
 * 配置广播接收者:
 *  <receiver android:name=".SMSBroadcastReceiver">
 *     <intent-filter android:priority="1000">
 *         <action android:name="android.provider.Telephony.SMS_RECEIVED"/>
 *     </intent-filter>
 *  </receiver>
 * 
 *  注意:
 *  <intent-filter android:priority="1000">表示:
 *  设置此广播接收者的级别为最高
 */

public class SMSBroadcastReceiver extends BroadcastReceiver {
    private static MessageListener mMessageListener;
    public SMSBroadcastReceiver() {
        super();
    }

    @Override
    public void onReceive(Context context, Intent intent) {
             Object [] pdus= (Object[]) intent.getExtras().get("pdus");
             for(Object pdu:pdus){
                SmsMessage smsMessage=SmsMessage.createFromPdu((byte [])pdu);
                String sender=smsMessage.getDisplayOriginatingAddress();
                String content=smsMessage.getMessageBody();
                long date=smsMessage.getTimestampMillis();
                Date timeDate=new Date(date);
                SimpleDateFormat simpleDateFormat=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
                String time=simpleDateFormat.format(timeDate);

                System.out.println("短信来自:"+sender);
                System.out.println("短信内容:"+content);
                System.out.println("短信时间:"+time);

                mMessageListener.OnReceived(content);

                //如果短信来自5556,不再往下传递,一般此号码可以作为短信平台的号码。
                if("5556".equals(sender)){
                    System.out.println(" abort ");
                    abortBroadcast();
                }

             }
    }

    // 回调接口
        public interface MessageListener {
            public void OnReceived(String message);
        }

        public void setOnReceivedMessageListener(MessageListener messageListener) {
            this.mMessageListener=messageListener;
        }
}

2.配置AndroidManifest.xml文件:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.messagecut"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="21" />

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name=".MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <receiver android:name=".SMSBroadcastReceiver">
          <intent-filter >
              <action android:name="android.provider.Telephony.SMS_RECEIVED"/>
          </intent-filter>
            </receiver>
    </application>

    <uses-permission android:name="android.permission.RECEIVE_SMS"/>
    <uses-permission android:name="android.permission.READ_SMS"/>

</manifest>

3.创建MainActivity.java-用于接收显示短信信息内容:

package com.example.messagecut;
import android.os.Bundle;
import android.widget.TextView;

import com.example.messagecut.SMSBroadcastReceiver.MessageListener;

import android.app.Activity;
/**
 * Demo描述:
 * 利用BroadcastReceiver实现监听短信
 *
 * 注意权限:
 * <uses-permission android:name="android.permission.RECEIVE_SMS"/>
 *
 * 详细资料:
 * http://blog.****.net/lfdfhl/article/details/8195400
 *
 */
public class MainActivity extends Activity{
    private TextView mTextView;
    private SMSBroadcastReceiver mSMSBroadcastReceiver;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        init();
    }
    private void init(){
        mTextView=(TextView) findViewById(R.id.textView);
        mSMSBroadcastReceiver=new SMSBroadcastReceiver();
        mSMSBroadcastReceiver.setOnReceivedMessageListener(new MessageListener() {
            public void OnReceived(String message) {
                mTextView.setText(message);
            }
        });
    }

}

4.布局文件很简单,TextView用于显示短信内容:

<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="com.example.messagecut.MainActivity" >

    <TextView
        android:id="@+id/textView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@string/hello_world" />

</RelativeLayout>

5.在真手机上运行项目,然后用另一台手机发送信息,得到如下截图:
Android实战简易课程-第三十六枪(监听短信-实现短信验证码自动填入)
截取到了短信内容。

二、截取验证码

上面实现了截取短信全部内容,下面我们看一下如何截取有效信息–验证码。原理应该很简单,就是在字符串中截取部分字符串。
算法即:

/**
       * 从字符串中截取连续6位数字组合 ([0-9]{" + 6 + "})截取六位数字 进行前后断言不能出现数字 用于从短信中获取动态密码
       * 
       * @param str
       *            短信内容
       * @return 截取得到的6位动态密码
       */
      public String getDynamicPassword(String str) {
        // 6是验证码的位数一般为六位
        Pattern continuousNumberPattern = Pattern.compile("(?<![0-9])([0-9]{"
            + 6 + "})(?![0-9])");
        Matcher m = continuousNumberPattern.matcher(str);
        String dynamicPassword = "";
        while (m.find()) {
          System.out.print(m.group());
          dynamicPassword = m.group();
        }

        return dynamicPassword;
      }

在MainActivity中:

package com.example.messagecut;
import android.os.Bundle;
import android.widget.EditText;
import android.widget.TextView;

import java.util.regex.Matcher;
import java.util.regex.Pattern;

import com.example.messagecut.SMSBroadcastReceiver.MessageListener;

import android.app.Activity;
/**
 * Demo描述:
 * 利用BroadcastReceiver实现监听短信
 *
 * 注意权限:
 * <uses-permission android:name="android.permission.RECEIVE_SMS"/>
 *
 * 
 *
 */
public class MainActivity extends Activity{
    private EditText mCode;
    private SMSBroadcastReceiver mSMSBroadcastReceiver;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        init();
    }
    private void init(){
        mCode=(EditText) findViewById(R.id.et_code);
        mSMSBroadcastReceiver=new SMSBroadcastReceiver();
        mSMSBroadcastReceiver.setOnReceivedMessageListener(new MessageListener() {
            public void OnReceived(String message) {
                mCode.setText(getDynamicPassword(message));//截取6位验证码
            }
        });
    }

    /**
       * 从字符串中截取连续6位数字组合 ([0-9]{" + 6 + "})截取六位数字 进行前后断言不能出现数字 用于从短信中获取动态密码
       * 
       * @param str
       *            短信内容
       * @return 截取得到的6位动态密码
       */
      public String getDynamicPassword(String str) {
        // 6是验证码的位数一般为六位
        Pattern continuousNumberPattern = Pattern.compile("(?<![0-9])([0-9]{"
            + 6 + "})(?![0-9])");
        Matcher m = continuousNumberPattern.matcher(str);
        String dynamicPassword = "";
        while (m.find()) {
          System.out.print(m.group());
          dynamicPassword = m.group();
        }

        return dynamicPassword;
      }

}

这里我们也修改了activity_main.xml文件,将TextView改成了EditText,因为android手机用户可能会禁止应用访问短信的权限应该支持手动填写。

在真机上测试,发送:
Android实战简易课程-第三十六枪(监听短信-实现短信验证码自动填入)
接收:
Android实战简易课程-第三十六枪(监听短信-实现短信验证码自动填入)
超吊超实用的功能,赶快引入到你的项目中去吧,会大大提高用户体验!

喜欢的朋友关注我吧!您的关注和支持是我的动力!

版权声明:本文为博主原创文章,未经博主允许不得转载。

4楼u010850027昨天 19:15
最近的项目正在研究这块的内容,感谢楼主的分享`(*∩_∩*)′
Re: yayun0516昨天 19:15
回复u010850027n多谢支持!
3楼yayun0516昨天 16:35
http://download.****.net/detail/yayun0516/9010671 源码下载
2楼yayun0516昨天 15:45
在mCode.setText(getDynamicPassword(message));//截取6位验证码 代码后面加上:mCode.setSelection(getDynamicPassword(message).length()); 可以在填充验证码后光标移动到最后面,这样感觉更好。
1楼u012232131昨天 10:18
思路很清晰,广播接收->提取MSG->通过接口回调给activity->截取验证码->显示到界面。赞!
Re: yayun0516昨天 10:27
回复u012232131n多谢支持!