阿里云+springboot 实现短信服务

 因为阿里云不能个人认证。

所以想要实现的也可以看看下面这篇文章。

http://t.csdn.cn/dBTqh

下面我还是通过阿里云为例。实现企业的方式

登录阿里云,点击产品,然后选择短信服务。点击国内信息

在这里,我们需要创建  签名。还有   模板

签名的名称最好根据你的公司名称起,或者自己随便起都行。

但是目前已经不允许个人申请了,就当了解,以后可能用到。

然后模板 ,就是你建好后,可以直接使用他的一个模板内容。比如下面图片所示。模板内容:

您的验证码为:${code},该验证码 5 分钟内有效,请勿泄漏于他人。

后面我们可以调用这个模板,传一个code(后台随机生成的验证码),就是可以输出,您的验证码为:${code},该验证码 5 分钟内有效,请勿泄漏于他人。给用户。

然后生成的模板,我们如果需要调用的话,就可以根据那个模板CODE 对应的编号,进行调用。

这是官方文档。

SendSms - 发送短信 (aliyun.com)

下面直接上代码

调用 AliMsgUtil.sendCode(phone, code),传入手机号和自定义验证码。(这是自己写的工具类,下面有)

 /**
     * 根据手机号获取验证码
     *
     * @param phone
     * @return
     */
    @RequestMapping(value = "/getCode.action", method = RequestMethod.GET)
    public MessageBean getCode(@NotBlank(message = "手机号不能为空") @RequestParam("phone") @Pattern(regexp = "^((13[0-9])|(14[5,7,9])|(15([0-3]|[5-9]))|(166)|(17[0,1,3,5,6,7,8])|(18[0-9])|(19[0-3]|[5-9]))\\d{8}$", message = "手机号码格式错误") String phone) {
        String code = RandomUtil.getRandomNum();
        System.out.println("验证码:" + code);
        try {

            if (AliMsgUtil.sendCode(phone, code)) {
                redisTemplateUtil.set(phone, code, 5 * 60);
            }
        } catch (ClientException e) {
            e.printStackTrace();
            return MessageUtil.failureBack();
        }
        return MessageUtil.success();
    }

 如果AliMsgUtil.sendCode(phone, code)返回true ,说明验证码已经成功发送给了用户。

这个时候,我们可以把验证码存储到 Redis上。

 static final String MODEL_CODE_SEND_CODE = "";
    static final String MODEL_CODE_SEND_MSG = "";    这里填写的是模板编号,也就是上面那种图片的 模板CODE

这个就是你控制台上的  KEY还有私钥

static final String ACCESS_KEY_ID = "";

    static final String ACCESS_KEY_SECRET = ""; 

request.setSignName("同城快递"); 这个就是你的签名,就是国内信息=>签名   的那个签名
package ;


import com.aliyuncs.DefaultAcsClient;
import com.aliyuncs.IAcsClient;
import com.aliyuncs.dysmsapi.model.v20170525.QuerySendDetailsRequest;
import com.aliyuncs.dysmsapi.model.v20170525.QuerySendDetailsResponse;
import com.aliyuncs.dysmsapi.model.v20170525.SendSmsRequest;
import com.aliyuncs.dysmsapi.model.v20170525.SendSmsResponse;
import com.aliyuncs.exceptions.ClientException;
import com.aliyuncs.profile.DefaultProfile;
import com.aliyuncs.profile.IClientProfile;

import java.text.SimpleDateFormat;
import java.util.Date;

/**
 * 阿里云短信
 */
public class AliMsgUtil {
    //产品名称:云通信短信API产品,开发者无需替换
    static final String PRODUCT = "Dysmsapi";
    //产品域名,开发者无需替换
    static final String DOMAIN = "dysmsapi.aliyuncs.com";

    // 此处需要替换成开发者自己的AK(在阿里云访问控制台寻找)
    static final String ACCESS_KEY_ID = "";

    static final String ACCESS_KEY_SECRET = "";

    static final String MODEL_CODE_SEND_CODE = "";
    static final String MODEL_CODE_SEND_MSG = "";

    /**
     * 阿里demo的方法
     *
     * @param mobile
     * @param json 验证码
     * @return
     * @throws ClientException
     */
    public static SendSmsResponse baseSend(String mobile, String json,String modelCode) throws ClientException {

        //可自助调整超时时间
        System.setProperty("sun.net.client.defaultConnectTimeout", "10000");
        System.setProperty("sun.net.client.defaultReadTimeout", "10000");
        //初始化acsClient,暂不支持region化
        IClientProfile profile = DefaultProfile.getProfile("cn-hangzhou", ACCESS_KEY_ID, ACCESS_KEY_SECRET);
        DefaultProfile.addEndpoint("cn-hangzhou", "cn-hangzhou", PRODUCT, DOMAIN);
        IAcsClient acsClient = new DefaultAcsClient(profile);
        
        //组装请求对象-具体描述见控制台-文档部分内容
        SendSmsRequest request = new SendSmsRequest();
        //必填:待发送手机号
        request.setPhoneNumbers(mobile);
        //必填:短信签名-可在短信控制台中找到
        request.setSignName("同城快递");
        //必填:短信模板-可在短信控制台中找到
        request.setTemplateCode(modelCode);
        //可选:模板中的变量替换JSON串,如模板内容为"亲爱的${name},您的验证码为${code}"时,此处的值为
        request.setTemplateParam(json);

        //选填-上行短信扩展码(无特殊需求用户请忽略此字段)
        //request.setSmsUpExtendCode("90997");

        //可选:outId为提供给业务方扩展字段,最终在短信回执消息中将此值带回给调用者
//        request.setOutId("yourOutId");

        //hint 此处可能会抛出异常,注意catch
        SendSmsResponse sendSmsResponse = acsClient.getAcsResponse(request);

        return sendSmsResponse;
    }

    /**
     * 发送验证码
     * 格式: 【同族汇】您的验证码为:1234,该验证码 5 分钟内有效,请勿泄漏于他人。
     * @param phone
     * @param code
     * @return
     * @throws ClientException
     */
    public static boolean sendCode(String phone,String code) throws ClientException {
        String json = "{\"code\":\""+code+"\"}";
        //发短信
        SendSmsResponse response = baseSend(phone,json,MODEL_CODE_SEND_CODE);
        System.out.println("短信接口返回的数据----------------");
        System.out.println("Code=" + response.getCode()+"; "+"Message=" + response.getMessage()+"; "+"RequestId=" + response.getRequestId()+"; "+"BizId=" + response.getBizId());
        //查明细
        if (response.getCode() != null && response.getCode().equals("OK")) {
            System.out.println("验证成功返回true");
            return true;
        }
        return false;
    }

    public static boolean msg(String phone ,Long orderId,String mobile) throws ClientException, InterruptedException {
        String json = "{\"order\":\""+orderId+"\",\"phone\":\""+mobile+"\"}";
        SendSmsResponse response = baseSend(phone,json,MODEL_CODE_SEND_MSG);
        System.out.println("短信接口返回的数据----------------");
        System.out.println("Code=" + response.getCode()+"; "+"Message=" + response.getMessage()+"; "+"RequestId=" + response.getRequestId()+"; "+"BizId=" + response.getBizId());
        //查明细
        if (response.getCode() != null && response.getCode().equals("OK")) {
//            QuerySendDetailsResponse querySendDetailsResponse = querySendDetails(response.getBizId(), phone);
//            System.out.println("短信明细查询接口返回数据-----------------");
//            System.out.println("Code=" + querySendDetailsResponse.getCode());
//            System.out.println("Message=" + querySendDetailsResponse.getMessage());
            return true;
        }
        return false;
    }
    /**
     * 查询短信
     * @param bizId
     * @param moblie
     * @return
     * @throws ClientException
     */
    public static QuerySendDetailsResponse querySendDetails(String bizId, String moblie) throws ClientException {

        //可自助调整超时时间
        System.setProperty("sun.net.client.defaultConnectTimeout", "10000");
        System.setProperty("sun.net.client.defaultReadTimeout", "10000");

        //初始化acsClient,暂不支持region化
        IClientProfile profile = DefaultProfile.getProfile("cn-hangzhou", ACCESS_KEY_ID, ACCESS_KEY_SECRET);
        DefaultProfile.addEndpoint("cn-hangzhou", "cn-hangzhou", PRODUCT, DOMAIN);
        IAcsClient acsClient = new DefaultAcsClient(profile);

        //组装请求对象
        QuerySendDetailsRequest request = new QuerySendDetailsRequest();
        //必填-号码
        request.setPhoneNumber(moblie);
        //可选-流水号
        request.setBizId(bizId);
        //必填-发送日期 支持30天内记录查询,格式yyyyMMdd
        SimpleDateFormat ft = new SimpleDateFormat("yyyyMMdd");
        request.setSendDate(ft.format(new Date()));
        //必填-页大小
        request.setPageSize(10L);
        //必填-当前页码从1开始计数
        request.setCurrentPage(1L);

        //hint 此处可能会抛出异常,注意catch
        QuerySendDetailsResponse querySendDetailsResponse = acsClient.getAcsResponse(request);

        return querySendDetailsResponse;
    }

//    /**
//     * @param mobile
//     * @param code 验证码
//     * @return
//     * @throws ClientException
//     * @throws InterruptedException
//     */
//    public static boolean sendMsgCode(String mobile, String code) throws ClientException, InterruptedException {
//        //发短信
        SendSmsResponse response = sendCode(mobile,code );
        System.out.println("短信接口返回的数据----------------");
        System.out.println("Code=" + response.getCode());
        System.out.println("Message=" + response.getMessage());
        System.out.println("RequestId=" + response.getRequestId());
        System.out.println("BizId=" + response.getBizId());
        Thread.sleep(3000L);
        //查明细
        if (response.getCode() != null && response.getCode().equals("OK")) {
            QuerySendDetailsResponse querySendDetailsResponse = querySendDetails(response.getBizId(), mobile);
            System.out.println("短信明细查询接口返回数据----------------");
            System.out.println("Code=" + querySendDetailsResponse.getCode());
            System.out.println("Message=" + querySendDetailsResponse.getMessage());
            return true;
        }
        return false;
//    }




}


 

然后当用户填写验证码时,就可以对比验证码是否正确。

 /**
     * 注册
     *
     * @param phone
     * @param code
     * @param loginPwd
     * @param confirmPwd
     * @return
     */
  @RequestMapping(value = "/register.action", method = RequestMethod.POST)
    public MessageBean userRegister(@NotBlank(message = "手机号不能为空") @Pattern(regexp = "^((13[0-9])|(14[5,7,9])|(15([0-3]|[5-9]))|(166)|(17[0,1,3,5,6,7,8])|(18[0-9])|(19[0-3]|[5-9]))\\d{8}$", message = "手机号码格式错误") String phone,

                                    @NotBlank(message = "验证码不能为空") String code,
                                    @NotBlank(message = "密码不能为空") String loginPwd,
                                    @NotBlank(message = "密码不能为空") String confirmPwd) throws UserExistException {
        if (!loginPwd.equals(confirmPwd)) {
            return MessageUtil.failure("两次输入的密码不一致");
        }

        String redisCode = (String) redisTemplateUtil.get(phone);
        if (!code.equals(redisCode)) {
            logger.info("code:" + code + "  redisCode: " + redisCode);
            return MessageUtil.failure("验证码错误");
        }

    
}

就这样,一个简单的用户获取验证码就完成了。快给我三连。

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
实现用户手机号验证码登录可以分为以下几个步骤: 1. 用户输入手机号和验证码,点击登录按钮。 2. 后端接收到手机号和验证码后,先验证验证码是否正确。 3. 如果验证码正确,后端生成JWT token并将token存储到Redis中,同时将token返回给前端。 4. 前端将token存储到本地,以便后续请求时使用。 5. 后续请求时,前端需要在请求头中加入token,后端通过解析token来判断用户是否已登录。 下面是具体实现过程: 1. 在阿里云短信控制台创建短信模板,获取accessKeyId和accessKeySecret。 2. 在Spring Boot项目中添加依赖: ``` <dependency> <groupId>com.aliyun</groupId> <artifactId>aliyun-java-sdk-core</artifactId> <version>4.0.3</version> </dependency> ``` 3. 实现发送短信验证码的接口: ``` @PostMapping("/sendSms") public Result sendSms(@RequestParam("phone") String phone) { // 生成随机验证码 String code = String.valueOf((int) ((Math.random() * 9 + 1) * 100000)); // 发送短信验证码 DefaultProfile profile = DefaultProfile.getProfile("cn-hangzhou", accessKeyId, accessKeySecret); IAcsClient client = new DefaultAcsClient(profile); CommonRequest request = new CommonRequest(); request.setSysMethod(MethodType.POST); request.setSysDomain("dysmsapi.aliyuncs.com"); request.setSysVersion("2017-05-25"); request.setSysAction("SendSms"); request.putQueryParameter("RegionId", "cn-hangzhou"); request.putQueryParameter("PhoneNumbers", phone); request.putQueryParameter("SignName", "短信签名"); request.putQueryParameter("TemplateCode", "短信模板编号"); request.putQueryParameter("TemplateParam", "{\"code\":\"" + code + "\"}"); try { CommonResponse response = client.getCommonResponse(request); // 将验证码存储到Redis中,有效期为5分钟 redisTemplate.opsForValue().set(phone, code, 5, TimeUnit.MINUTES); return Result.success("短信验证码发送成功"); } catch (Exception e) { return Result.error("短信验证码发送失败"); } } ``` 4. 实现用户手机号验证码登录的接口: ``` @PostMapping("/login") public Result login(@RequestParam("phone") String phone, @RequestParam("code") String code) { // 验证验证码是否正确 String redisCode = redisTemplate.opsForValue().get(phone); if (StringUtils.isBlank(redisCode)) { return Result.error("验证码已过期,请重新发送"); } if (!redisCode.equals(code)) { return Result.error("验证码不正确"); } // 生成JWT token,并存储到Redis中 String token = JwtUtils.generateToken(phone); redisTemplate.opsForValue().set(phone, token, 1, TimeUnit.DAYS); // 将token返回给前端 return Result.success(token); } ``` 5. 实现JWT token的生成和解析: ``` public class JwtUtils { private static final String SECRET_KEY = "jwt_secret_key"; // JWT密钥 private static final long EXPIRATION_TIME = 7 * 24 * 60 * 60 * 1000; // JWT过期时间(7天) public static String generateToken(String phone) { Date now = new Date(); Date expiration = new Date(now.getTime() + EXPIRATION_TIME); return Jwts.builder() .setSubject(phone) .setIssuedAt(now) .setExpiration(expiration) .signWith(SignatureAlgorithm.HS256, SECRET_KEY) .compact(); } public static String getPhoneFromToken(String token) { try { Claims claims = Jwts.parser().setSigningKey(SECRET_KEY).parseClaimsJws(token).getBody(); return claims.getSubject(); } catch (Exception e) { return null; } } } ``` 6. 在拦截器中验证token并获取用户信息: ``` public class JwtInterceptor implements HandlerInterceptor { private static final String AUTH_HEADER = "Authorization"; // token在请求头中的名称 @Autowired private StringRedisTemplate redisTemplate; @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { String token = request.getHeader(AUTH_HEADER); if (StringUtils.isBlank(token)) { throw new BusinessException("未登录或登录已过期"); } String phone = JwtUtils.getPhoneFromToken(token); if (StringUtils.isBlank(phone)) { throw new BusinessException("无效的token"); } String redisToken = redisTemplate.opsForValue().get(phone); if (StringUtils.isBlank(redisToken) || !redisToken.equals(token)) { throw new BusinessException("未登录或登录已过期"); } return true; } } ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值