Maven依赖
<dependency>
<groupId>com.aliyun</groupId>
<artifactId>aliyun-java-sdk-core</artifactId>
<version>4.0.6</version> <!-- 注:如提示报错,先升级基础包版,无法解决可联系技术支持 -->
</dependency>
<dependency>
<groupId>com.aliyun</groupId>
<artifactId>aliyun-java-sdk-dysmsapi</artifactId>
<version>1.1.0</version>
</dependency>
1.去阿里云注册账号后创建短信模板和短信签名(在后边需要用到短信签名以及短信模板版)
2.创建AccessKeyId
3.代码实现发送短信
import com.aliyuncs.DefaultAcsClient;
import com.aliyuncs.IAcsClient;
import com.aliyuncs.dysmsapi.model.v20170525.SendSmsRequest;
import com.aliyuncs.dysmsapi.model.v20170525.SendSmsResponse;
import com.aliyuncs.exceptions.ClientException;
import com.aliyuncs.http.MethodType;
import com.aliyuncs.profile.DefaultProfile;
import com.aliyuncs.profile.IClientProfile;
public class SendSMSUtil {
// 替换成你的AK
private static final String accessKeyId = "*********";// 你的accessKeyId
private static final String accessKeySecret = "***********";// 你的accessKeySecret
private static final String signName = "*********";// 签名
private static final String templateCode = "**********";// 短信模板
private static int code;
/**
* @Description:发送手机验证码
* @Param:需要发送的手机号码
* @return:OK表示成功,失败则返回失败信息
*/
public String senSMSUtil(String phoneNumber) {
// 设置超时时间-可自行调整
System.setProperty("sun.net.client.defaultConnectTimeout", "10000");
System.setProperty("sun.net.client.defaultReadTimeout", "10000");
// 初始化ascClient需要的几个参数
final String product = "Dysmsapi";// 短信API产品名称(短信产品名固定,无需修改)
final String domain = "dysmsapi.aliyuncs.com";// 短信API产品域名(接口地址固定,无需修改)
// 初始化ascClient,暂时不支持多region(请勿修改)
IClientProfile profile = DefaultProfile.getProfile("cn-hangzhou", accessKeyId, accessKeySecret);
try {
DefaultProfile.addEndpoint("cn-hangzhou", "cn-hangzhou", product, domain);
} catch (ClientException e) {
e.printStackTrace();
}
IAcsClient acsClient = new DefaultAcsClient(profile);
// 组装请求对象
SendSmsRequest request = new SendSmsRequest();
// 使用post提交
request.setMethod(MethodType.POST);
// 必填:待发送手机号。支持以逗号分隔的形式进行批量调用,批量上限为1000个手机号码,批量调用相对于单条调用及时性稍有延迟,
// 验证码类型的短信推荐使用单条调用的方式;发送国际/港澳台消息时,接收号码格式为国际区号+号码,如“85200000000”
request.setPhoneNumbers(phoneNumber);
request.setSignName(signName);
// 必填:短信模板-可在短信控制台中找到,发送国际/港澳台消息时,请使用国际/港澳台短信模版
request.setTemplateCode(templateCode);
//随机生成六位验证码
code = (int) ((Math.random() * 9 + 1) * 100000);
// 可选:模板中的变量替换JSON串,如模板内容为"亲爱的${name},您的验证码为${code}"时,此处的值为
// 友情提示:如果JSON中需要带换行符,请参照标准的JSON协议对换行符的要求,比如短信内容中包含\r\n的情况在JSON中需要表示成\\r\\n,否则会导致JSON在服务端解析失败
request.setTemplateParam("{code:" + code + "}");
// 请求失败这里会抛ClientException异常
SendSmsResponse sendSmsResponse = null;
try {
sendSmsResponse = acsClient.getAcsResponse(request);
} catch (ClientException e) {
e.printStackTrace();
return "请求失败";
}
assert sendSmsResponse.getCode() != null;
if (sendSmsResponse.getCode() == null || !sendSmsResponse.getCode().equals("OK")) {// 发送不成功
return sendSmsResponse.getMessage();
}
// 请求成功
return "OK";
}
public int getCode() {
return code;
}
}
4.完善发送短信接口
/**
* @Description:发送手机验证码
* @Param:手机号码
* @return:1表示成功,0表示失败
*/
@RequestMapping(value = "/SendSMS", method = RequestMethod.POST)
@ResponseBody
public String SMSTest(String phoneNumber, HttpServletRequest request) {
// 发送短信
SendSMSUtil sendSMS = new SendSMSUtil();
String result = sendSMS.senSMSUtil(phoneNumber);
if (result == null || !result.equals("OK")) {// 发送不成功
return "0";
}
// 获取验证码
int code = sendSMS.getCode();
// 将数据存入SESSION
request.getSession().setAttribute("checkCode", code);
// 将验证码生成时间存入SESSION,若超过五分钟则不通过校验
request.getSession().setAttribute("createTime", System.currentTimeMillis());
return "OK";
}
5.校验验证码接口
/**
* @Description:校验验证码是否正确
* @Param:验证码
* @return:成功返回OK,验证码超时返回TimeOut,验证码错误返回CodeError
*/
@RequestMapping(value = "/checkSMSCode", method = RequestMethod.POST)
@ResponseBody
public String checkSMSCode(String checkSMSCode, HttpServletRequest request) {
// 服务器放入的验证码
String serverCheckCode = request.getSession().getAttribute("checkCode").toString();
// 验证码创建时间
String serverCreateTime = request.getSession().getAttribute("createTime").toString();
// 当前时间
long uCreateTime = Long.parseLong(String.valueOf(System.currentTimeMillis()));
if (serverCheckCode == null || serverCreateTime == null || serverCheckCode.equals("") || serverCreateTime.equals("")) {
return "CodeError";
}
if (!checkSMSCode.equals(serverCheckCode)) {// 验证码不匹配
return "CodeError";
}
long sCreateTime = Long.parseLong(serverCreateTime);// 验证码创建时间
if (uCreateTime - sCreateTime >= 300000) {// 如果时间大于五分钟
return "TimeOut";
}
return "OK";
}
6.测试
注意查看浏览器中的地址,发送与校验验证码均是返回OK,此时则为登录成功。
附上测试接口的jsp页面代码
<%--
Created by IntelliJ IDEA.
User: GuiH
Date: 2019-04-24
Time: 11:33
To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
login page
<fieldset>
<form action="/SendSMS" method="post">
手机号:<input type="text" name="phoneNumber">
<input type="submit" value="发送">
</form>
<form action="/checkSMSCode" method="post">
验证码:<input type="text" name="checkSMSCode">
<input type="submit" value="提交">
</form>
</fieldset>
</body>
</html>
到此短信验证码功能就完成了,此处使用的是使用会话中的SESSION来完成的短信验证码,还可以尝试使用ehcache缓存框架来实现。