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.去阿里云注册账号后创建短信模板和短信签名(在后边需要用到短信签名以及短信模板版)

JAVA短信验证码注册 java短信验证码实现流程_短信验证码

JAVA短信验证码注册 java短信验证码实现流程_短信接口_02


2.创建AccessKeyId

JAVA短信验证码注册 java短信验证码实现流程_短信验证码_03




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.测试

JAVA短信验证码注册 java短信验证码实现流程_Java短信验证码_04


JAVA短信验证码注册 java短信验证码实现流程_短信接口_05

注意查看浏览器中的地址,发送与校验验证码均是返回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缓存框架来实现。