【mob】Android短信验证+源码

时间:2023-03-08 19:58:44
【mob】Android短信验证+源码

在很多的应用当中,都涉及到了短信验证的功能,比如在注册或者找回密码的时候,那么我们如何通过第三方的平台来完成这个功能呢?

本面博文就实现短信验证,来做一个小的栗子。

第一步-下载开发包

首先你要在第三方平台mob拥有一个开发者账号,这样你才能使用其提供的短信验证服务。


【mob】Android短信验证+源码


然后点击下载相应开发平台的版本,我这里使用的是AS。


【mob】Android短信验证+源码


下载完毕后我们得到的是一个压缩包,解压缩,打开SMSSDK这个文件夹。


【mob】Android短信验证+源码


建议大家可以先看看HowToUse里面的内容:


【mob】Android短信验证+源码


第二步-将SDK导入到项目当中

官方的文档说明,2.0.1开始使用aar文件集成。更方便快捷。所以这里建议各位将自己的AS升级到2.0.1的版本之上,这样AS才能识别.aar后缀的文件。

把SMSSDK中的文件放在Module所在的Libs里面,如图。 
【mob】Android短信验证+源码

然后子啊该Module的build.gradle(上图高亮的build.gradle,不是整个项目的build.gradle)当中添加依赖。

【mob】Android短信验证+源码

这样一来,前期的准备工作就都结束了。接下来的工作就是去调用开发包中的各种方法。

第三步-启动SDK

要想使用SDK提供的各种功能,首先就必须要启动SDk。

首先我们要进入到刚才注册过的mob官网,然后点击【进入后台】


【mob】Android短信验证+源码


里面有mob的四个主要服务,【ShareSDK】、【SecurityCodeSDK】、【ShareREC】、【MobAPI】,我们要使用的【SecurityCodeSDK】。


【mob】Android短信验证+源码


接着,我们要创建一个应用。名称就填写我们刚才在AS中创建的项目名称即可。


【mob】Android短信验证+源码


最关键的内容,就是你的App Key和App secret


【mob】Android短信验证+源码


把它们复制下来,放到AS当中。

【mob】Android短信验证+源码

接下来就可以启动短信验证的SDK了。

 // 启动短信验证sdk
SMSSDK.initSDK(this, appKey, appSecret);

第四部-注册短信回调

这里为什么要注册短信回调呢?我的理解是,这里的工作原理和广播非常相似,我们在使用广播的时候需要注册一个广播接收器,以便对不同的消息做出反义。

注册的方法如下: 
(1)定义一个EventHandler

 private EventHandler eh;

(2)编写EventHandler的事件处理 
EventHandler即为操作回调。它包括4个方法,分别为:

public void onRegister();
public void beforeEvent(int event, Object data);
public void afterEvent(int event, int result, Object data);
public void onUnregister();

其中onRegister在回调对象注册的时候被触发。beforeEvent在操作执行前被触发,其参数event表示操作的类型,data是从外部传入的数据。afterEvent在操作结束时被触发,同样具备event和data参数,但是data是事件操作结果,其具体取值根据参数result而定。result是操作结果,为SMSSDK.RESULT_COMPLETE表示操作成功,为SMSSDK.RESULT_ERROR表示操作失败。

更多详细的内容,大家可以去看一下官网的这篇文章——Android 短信SDK操作回调

根据官方文档提供的几个Result值,我们对afterEvent中的收到的几种消息做出处理,代码如下:

 eh=new EventHandler(){
@Override
public void afterEvent(int event, int result, Object data) {
if (result == SMSSDK.RESULT_COMPLETE) {
//回调完成
if (event == SMSSDK.EVENT_SUBMIT_VERIFICATION_CODE) {
//提交验证码成功
Message msg = new Message();
msg.arg1 = 0;
msg.obj = data;
handler.sendMessage(msg);
Log.d(TAG, "提交验证码成功");
} else if (event == SMSSDK.EVENT_GET_VERIFICATION_CODE) {
Message msg = new Message();
//获取验证码成功
msg.arg1 = 1;
msg.obj = "获取验证码成功";
handler.sendMessage(msg);
Log.d(TAG, "获取验证码成功");
} else if (event == SMSSDK.EVENT_GET_SUPPORTED_COUNTRIES) {
Message msg = new Message();
//返回支持发送验证码的国家列表
msg.arg1 = 2;
msg.obj = "返回支持发送验证码的国家列表";
handler.sendMessage(msg);
Log.d(TAG, "返回支持发送验证码的国家列表");
}
} else {
Message msg = new Message();
//返回支持发送验证码的国家列表
msg.arg1 = 3;
msg.obj = "验证失败";
handler.sendMessage(msg);
Log.d(TAG, "验证失败");
((Throwable) data).printStackTrace();
}
}
};

由于EventHandler开启了线程,所以不能直接在afterEvent中更新UI,所以还需要在MainActivity当中定义一个Handler来接受EventHandler发送过来的消息。

   private Handler handler=new Handler(){
@Override
public void handleMessage(Message msg) {
switch (msg.arg1) {
case 0:
//客户端验证成功,可以进行注册,返回校验的手机和国家代码phone/country
Toast.makeText(MainActivity.this, msg.obj.toString(), Toast.LENGTH_SHORT).show();
break;
case 1:
//获取验证码成功
Toast.makeText(MainActivity.this, msg.obj.toString(), Toast.LENGTH_SHORT).show();
break;
case 2:
//返回支持发送验证码的国家列表
Toast.makeText(MainActivity.this, msg.obj.toString(), Toast.LENGTH_SHORT).show();
break;
}
}
};

完成上述操作之后,注册短信回调

SMSSDK.registerEventHandler(eh); //注册短信回调
  • 1
  • 1

第五部-完善UI

主界面布局如下

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"> <EditText
android:id="@+id/et_phone"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:hint="请输入手机号" /> <Button
android:id="@+id/bt_getCode"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="获取验证码" /> <EditText
android:id="@+id/et_code"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:hint="输入验证码"/> <Button
android:id="@+id/bt_verify"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="验证"/> </LinearLayout>

非常简单,et_phone用于输入手机号,bt_getCode获取验证码,et_code用于输入手机收到的验证码,bt_vertify对验证码进行验证。

Java部分的代码重点讲下为按钮bt_getCode和bt_vertify设置监听事件。

 bt_getCode.setClickable(false);
bt_getCode.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//获取验证码操作
phone=((EditText)findViewById(R.id.et_phone)).getText().toString();
if(phone.equals("")){
Toast.makeText(MainActivity.this,"手机号不能为空",Toast.LENGTH_SHORT).show();
}else{
//填写了手机号码,isMobileNO(phone)判断号码格式
if(isMobileNO(phone)){
//如果手机号码无误,则发送验证请求
bt_getCode.setClickable(true);
//让按钮的样式变成60S倒计时
changeBtnGetCode();
//SMSSDK中自带的2个方法
getSupportedCountries();
getVerificationCode("86", phone);
}else{
//手机号格式有误
Toast.makeText(MainActivity.this,"手机号格式错误,请检查",Toast.LENGTH_SHORT).show();
}
}
}
});

其中 changeBtnGetCode();的代码为:

   /*
* 改变按钮样式
* */
private void changeBtnGetCode() { Thread thread = new Thread() {
@Override
public void run() {
if (tag) {
while (i > 0) {
i--;
//如果活动为空
if (MainActivity.this == null) {
break;
} MainActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
bt_getCode.setText("获取验证码(" + i + ")");
bt_getCode.setClickable(false);
}
}); try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
tag = false;
}
i = 60;
tag = true; if (MainActivity.this != null) {
MainActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
bt_getCode.setText("获取验证码");
bt_getCode.setClickable(true);
}
});
}
}
};
thread.start();
}

isMobileNO()的代码为:

    private boolean isMobileNO(String phone) {
/*
移动:134、135、136、137、138、139、150、151、157(TD)、158、159、187、188
联通:130、131、132、152、155、156、185、186
电信:133、153、180、189、(1349卫通)
总结起来就是第一位必定为1,第二位必定为3或5或8,其他位置的可以为0-9
*/
String telRegex = "[1][358]\\d{9}";//"[1]"代表第1位为数字1,"[358]"代表第二位可以为3、5、8中的一个,"\\d{9}"代表后面是可以是0~9的数字,有9位。
if (TextUtils.isEmpty(phone)) return false;
else return phone.matches(telRegex);
}
}

然后是 bt_vertify的监听时间,比较简单

  bt_vertify.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//验证操作
code=((EditText)findViewById(R.id.et_code)).getText().toString();
if (code.equals("")){
Toast.makeText(MainActivity.this,"验证码不能为空",Toast.LENGTH_SHORT).show();
}else{
//填写了验证码,进行验证,SMSSDK自带的方法
submitVerificationCode("86", phone, code);
}
}
});

最后别忘记在AndroidManifest.xml中添加网络访问的权限

   <uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />

附录

效果图

【mob】Android短信验证+源码

完整JAVA代码

import android.os.Handler;
import android.os.Message;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.text.TextUtils;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.Toast; import cn.smssdk.EventHandler;
import cn.smssdk.SMSSDK; import static cn.smssdk.SMSSDK.getSupportedCountries;
import static cn.smssdk.SMSSDK.getVerificationCode;
import static cn.smssdk.SMSSDK.submitVerificationCode; public class MainActivity extends AppCompatActivity {
private final String TAG="--MainActivity--";
//app key和app secret 需要填自己应用的对应的!这里只是我自己创建的应用。
private final String appKey="1549a02f213cb";
private final String appSecret="5753971a3b122dd9caf36a23a59ba5d9";
private EventHandler eh;
private Handler handler=new Handler(){
@Override
public void handleMessage(Message msg) {
switch (msg.arg1) {
case 0:
//客户端验证成功,可以进行注册,返回校验的手机和国家代码phone/country
Toast.makeText(MainActivity.this, msg.obj.toString(), Toast.LENGTH_SHORT).show();
break;
case 1:
//获取验证码成功
Toast.makeText(MainActivity.this, msg.obj.toString(), Toast.LENGTH_SHORT).show();
break;
case 2:
//返回支持发送验证码的国家列表
Toast.makeText(MainActivity.this, msg.obj.toString(), Toast.LENGTH_SHORT).show();
break;
}
}
};
//View控件
private Button bt_getCode;
private Button bt_vertify;
//手机号码
private String phone;
//验证码
private String code; private boolean isChange;
//控制按钮样式是否改变
private boolean tag = true;
//每次验证请求需要间隔60S
private int i=60; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main_activity); // 启动短信验证sdk
SMSSDK.initSDK(this, appKey, appSecret); eh=new EventHandler(){
@Override
public void afterEvent(int event, int result, Object data) {
if (result == SMSSDK.RESULT_COMPLETE) {
//回调完成
if (event == SMSSDK.EVENT_SUBMIT_VERIFICATION_CODE) {
//提交验证码成功
Message msg = new Message();
msg.arg1 = 0;
msg.obj = data;
handler.sendMessage(msg);
Log.d(TAG, "提交验证码成功");
} else if (event == SMSSDK.EVENT_GET_VERIFICATION_CODE) {
Message msg = new Message();
//获取验证码成功
msg.arg1 = 1;
msg.obj = "获取验证码成功";
handler.sendMessage(msg);
Log.d(TAG, "获取验证码成功");
} else if (event == SMSSDK.EVENT_GET_SUPPORTED_COUNTRIES) {
Message msg = new Message();
//返回支持发送验证码的国家列表
msg.arg1 = 2;
msg.obj = "返回支持发送验证码的国家列表";
handler.sendMessage(msg);
Log.d(TAG, "返回支持发送验证码的国家列表");
}
} else {
Message msg = new Message();
//返回支持发送验证码的国家列表
msg.arg1 = 3;
msg.obj = "验证失败";
handler.sendMessage(msg);
Log.d(TAG, "验证失败");
((Throwable) data).printStackTrace();
}
}
}; SMSSDK.registerEventHandler(eh); //注册短信回调 bt_getCode= (Button) findViewById(R.id.bt_getCode);
bt_getCode.setClickable(false);
bt_vertify= (Button) findViewById(R.id.bt_verify);
bt_getCode.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//获取验证码操作
phone=((EditText)findViewById(R.id.et_phone)).getText().toString();
if(phone.equals("")){
Toast.makeText(MainActivity.this,"手机号不能为空",Toast.LENGTH_SHORT).show();
}else{
//填写了手机号码
if(isMobileNO(phone)){
//如果手机号码无误,则发送验证请求
bt_getCode.setClickable(true);
changeBtnGetCode();
getSupportedCountries();
getVerificationCode("86", phone);
}else{
//手机号格式有误
Toast.makeText(MainActivity.this,"手机号格式错误,请检查",Toast.LENGTH_SHORT).show();
}
}
}
}); bt_vertify.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//验证操作
code=((EditText)findViewById(R.id.et_code)).getText().toString();
if (code.equals("")){
Toast.makeText(MainActivity.this,"验证码不能为空",Toast.LENGTH_SHORT).show();
}else{
//填写了验证码,进行验证
submitVerificationCode("86", phone, code);
}
}
});
} /*
* 改变按钮样式
* */
private void changeBtnGetCode() { Thread thread = new Thread() {
@Override
public void run() {
if (tag) {
while (i > 0) {
i--;
//如果活动为空
if (MainActivity.this == null) {
break;
} MainActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
bt_getCode.setText("获取验证码(" + i + ")");
bt_getCode.setClickable(false);
}
}); try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
tag = false;
}
i = 60;
tag = true; if (MainActivity.this != null) {
MainActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
bt_getCode.setText("获取验证码");
bt_getCode.setClickable(true);
}
});
}
}
};
thread.start();
} private boolean isMobileNO(String phone) {
/*
移动:134、135、136、137、138、139、150、151、157(TD)、158、159、187、188
联通:130、131、132、152、155、156、185、186
电信:133、153、180、189、(1349卫通)
总结起来就是第一位必定为1,第二位必定为3或5或8,其他位置的可以为0-9
*/
String telRegex = "[1][358]\\d{9}";//"[1]"代表第1位为数字1,"[358]"代表第二位可以为3、5、8中的一个,"\\d{9}"代表后面是可以是0~9的数字,有9位。
if (TextUtils.isEmpty(phone)) return false;
else return phone.matches(telRegex);
}
}

源码下载

项目地址:github

说明

1、为了方便大家的导入,这里附上我的AS的相关信息 
【mob】Android短信验证+源码 
【mob】Android短信验证+源码 
大家可以根据自己的开发环境进行配置

2、这个栗子只是简单的调用了SMSSDK的相关方法,在实际开发的过程中,需要结合项目的需求,活学活用