您的位置:首页 > 移动开发 > Android开发

Android短信验证+源码

2016-07-23 17:31 513 查看
本人水平有限,文章中如果出现什么不正确或者模糊的地方,还请各位小伙伴留下评论,多多指教 : )

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

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

第一步-下载开发包

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

第三步-启动SDK

第四步-注册短信回调

第五部-完善UI

附录
效果图

完整JAVA代码

源码下载

说明

第一步-下载开发包

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



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



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



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



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

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

把SMSSDK中的文件放在Module所在的Libs里面,如图。



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



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

第三步-启动SDK

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

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



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



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



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



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



接下来就可以启动短信验证的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); //注册短信回调


第五部-完善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" />


附录

效果图



完整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的相关信息





大家可以根据自己的开发环境进行配置

2、这个栗子只是简单的调用了SMSSDK的相关方法,在实际开发的过程中,需要结合项目的需求,活学活用。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  android 短信