springboot整合redis之發送手機驗證碼


阿里雲服務發送手機短信驗證碼-----(第二篇)

文章概述:springboot整合redis之發送手機驗證碼注冊登錄

注:搭建springboot項目可以參考這篇文章:

前言:短信驗證碼是通過發送驗證碼到手機的一種有效的驗證碼系統。主要用於驗證用戶手機的合法性及敏感操作的身份驗證。常見的使用場景有:登錄注冊、信息修改、異常登錄、找回密碼等操作。
用戶注冊發送驗證碼,然后核實對比用戶注冊成功采用redis方式將手機號碼+key放入redis緩存中設置驗證碼超時時間,比對用戶名和驗證碼采用數據庫存儲方式,注冊時拿取redis中驗證碼進行判讀驗證碼是否過期是否匹配。

1.pom.xml文件配置中添加以下依賴

  <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>

        <!-- 短信包-->
        <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>

        <!--redis依賴-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-redis</artifactId>
        </dependency>
    </dependencies>

2.application.yml添加如下配置

# 服務端口
server:
  port: 8080

spring:
#reids配置
redis:
  key:
    prefix:
      authCode: "portal:authCode:"
      orderId: "portal:orderId:"
    expire:
      authCode: 60 # 驗證碼超期時間
  host: 127.0.0.1 # Redis服務器地址
  database: 1 # Redis數據庫索引(默認為0)
  port: 6379 # Redis服務器連接端口
  password: # Redis服務器連接密碼(默認為空)
  jedis:
    pool:
      max-active: 8 # 連接池最大連接數(使用負值表示沒有限制)
      max-wait: -1ms # 連接池最大阻塞等待時間(使用負值表示沒有限制)
      max-idle: 8 # 連接池中的最大空閑連接
      min-idle: 0 # 連接池中的最小空閑連接
  timeout: 3000ms # 連接超時時間(毫秒)

3.生成短信隨機驗證碼工具類

import java.util.Random;

public class CodeUtil {

    //使用到Algerian字體,系統里沒有的話需要安裝字體,字體只顯示大寫,去掉了1,0,i,o幾個容易混淆的字符
    public static final String VERIFY_CODES = "1234567890";

    /**
     * 使用系統默認字符源生成驗證碼
     * @param verifySize    驗證碼長度
     * @return
     */
    public static String generateVerifyCode(int verifySize){
        return generateVerifyCode(verifySize, VERIFY_CODES);
    }

    /**
     * 使用指定源生成驗證碼
     * @param verifySize    驗證碼長度
     * @param sources   驗證碼字符源
     * @return
     */
    public static String generateVerifyCode(int verifySize, String sources){
        if(sources == null || sources.length() == 0){
            sources = VERIFY_CODES;
        }
        int codesLen = sources.length();
        Random rand = new Random(System.currentTimeMillis());
        StringBuilder verifyCode = new StringBuilder(verifySize);
        for(int i = 0; i < verifySize; i++){
            verifyCode.append(sources.charAt(rand.nextInt(codesLen-1)));
        }
        return verifyCode.toString();
    }

    public static void main(String[] args) {
        System.out.println(generateVerifyCode(4));
    }
}

4.短信下發工具類

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.profile.DefaultProfile;
import com.aliyuncs.profile.IClientProfile;

public class SmsTool {

    //產品名稱:雲通信短信API產品,開發者無需替換
    static final String product = "Dysmsapi";
    //產品域名,開發者無需替換
    static final String domain = "dysmsapi.aliyuncs.com";
    // TODO 此處需要替換成開發者自己的AK(在阿里雲訪問控制台尋找)
    static final String accessKeyId = "自己的AccessKeyId";
    static final String accessKeySecret = "自己的accesskeySecret";

    public static SendSmsResponse sendSms(String phone , String code) throws ClientException {
        //可自助調整超時時間
        System.setProperty("sun.net.client.defaultConnectTimeout", "10000");
        System.setProperty("sun.net.client.defaultReadTimeout", "10000");
        //初始化acsClient,暫不支持region化
        IClientProfile profile = DefaultProfile.getProfile("cn-hangzhou", accessKeyId, accessKeySecret);
        DefaultProfile.addEndpoint("cn-hangzhou", "cn-hangzhou", product, domain);
        IAcsClient acsClient = new DefaultAcsClient(profile);

        //組裝請求對象-具體描述見控制台-文檔部分內容
        SendSmsRequest request = new SendSmsRequest();
        //必填:待發送手機號
        request.setPhoneNumbers(phone);
        //必填:短信簽名-可在短信控制台中找到
        request.setSignName("替換成自己的短信簽名");
        //必填:短信模板-可在短信控制台中找到
        request.setTemplateCode("替換成自己的短信模板編號");
        //可選:模板中的變量替換JSON串,如模板內容為"親愛的${name},您的驗證碼為${code}"時,此處的值為
        request.setTemplateParam(code);

        //選填-上行短信擴展碼(無特殊需求用戶請忽略此字段)
        //request.setSmsUpExtendCode("90997");

        //可選:outId為提供給業務方擴展字段,最終在短信回執消息中將此值帶回給調用者
        //request.setOutId("yourOutId");

        //hint 此處可能會拋出異常,注意catch
        SendSmsResponse sendSmsResponse = acsClient.getAcsResponse(request);
        return sendSmsResponse;
    }
}

5.reids工具類接口

public interface RedisService {

    /**
     * 存儲數據
     */
    void set(String key, String value);

    /**
     * 獲取數據
     */
    String get(String key);

    /**
     * 設置超期時間
     */
    boolean expire(String key, long expire);

    /**
     * 刪除數據
     */
    void remove(String key);

    /**
     * 自增操作
     * @param delta 自增步長
     */
    Long increment(String key, long delta);

}

6.reids實現類

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.stereotype.Service;

import java.util.concurrent.TimeUnit;

@Service
public class RedisServiceImpl implements  RedisService{
    @Autowired
    private StringRedisTemplate stringRedisTemplate;

    @Override
    public void set(String key, String value) {
        stringRedisTemplate.opsForValue().set(key, value);
    }

    @Override
    public String get(String key) {
        return stringRedisTemplate.opsForValue().get(key);
    }

    @Override
    public boolean expire(String key, long expire) {
        return stringRedisTemplate.expire(key, expire, TimeUnit.SECONDS);
    }

    @Override
    public void remove(String key) {
        stringRedisTemplate.delete(key);
    }

    @Override
    public Long increment(String key, long delta) {
        return stringRedisTemplate.opsForValue().increment(key,delta);
    }

}

7.Controller攔截類

import com.aliyuncs.dysmsapi.model.v20170525.SendSmsResponse;
import com.aliyuncs.exceptions.ClientException;
import com.aliyuncs.utils.StringUtils;
import or.og.smsdemo.redisutil.RedisService;
import or.og.smsdemo.util.CodeUtil;
import or.og.smsdemo.util.SmsTool;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.*;

import javax.servlet.http.HttpServletRequest;
import java.util.HashMap;
import java.util.Map;

@Controller
@RequestMapping("/sms")
public class SmsCtrlController {


    /***
     * 注入redis模版
     */
    @Autowired
    private RedisService redisService;
    private  String tokenId="TOKEN-USER-";


    /**
     * 發送短信
     * @ResponseBody 返回json數據
     * @RequestMapping 攔截請求,指定請求類型:POST
     * @RequestBody 接受前台傳入的json數據 接受類型為Map
     * @throws ClientException 拋出異常
     */
    @ResponseBody
    @RequestMapping(value = "/smsXxs", method = RequestMethod.POST, headers = "Accept=application/json")
    public Map<String,Object> smsXxs(@RequestBody Map<String,Object> requestMap,HttpServletRequest request) throws ClientException {
        Map<String,Object> map = new HashMap<>();
        String phone = requestMap.get("phoneNumber").toString();
        // 調用工具欄中生成驗證碼方法(指定長度的隨機數)
        String code = CodeUtil.generateVerifyCode(6);
        //填充驗證碼
        String TemplateParam = "{\"code\":\""+code+"\"}";
        SendSmsResponse response = SmsTool.sendSms(phone,TemplateParam);//傳入手機號碼及短信模板中的驗證碼占位符

        map.put("verifyCode",code);
        map.put("phone",phone);
        request.getSession().setAttribute("CodePhone",map);
        if( response.getCode().equals("OK")) {
            map.put("isOk","OK");
            //驗證碼綁定手機號並存儲到redis
            redisService.set(tokenId+phone,code);
            redisService.expire(tokenId+phone,620);//調用reids工具類中存儲方法設置超時時間
        }
        return map;
    }


    /**
     * 注冊驗證
     * @ResponseBody 返回json數據
     * @RequestMapping 攔截請求,指定請求類型:POST
     * @RequestBody 接受前台傳入的json數據 接受類型為Map
     * @throws ClientException 拋出異常
     */
    @ResponseBody
    @RequestMapping(value = "/validateNum", method = RequestMethod.POST, headers = "Accept=application/json")
    public Map<String, Object> validateNum(@RequestBody Map<String,Object> requestMap) throws ClientException {

        Map<String,Object> map = new HashMap<>();
        String phone = requestMap.get("phone").toString();//獲取注冊手機號碼
        String verifyCode = requestMap.get("verifyCode").toString();//獲取手機驗證碼

        //首先比對驗證碼是否失效
        String redisauthcode= redisService.get(tokenId+phone); //傳入tonkenId返回redis中的value
        if(StringUtils.isEmpty(redisauthcode)){
            //如果未取到則過期驗證碼已失效
            map.put("ruselt",404);
        }else  if(!"".equals(redisauthcode)&&!verifyCode.equals(redisauthcode)){
            //驗證碼錯誤
            map.put("ruselt",500);
        }else{
            //用戶注冊成功
            map.put("ruselt",200);
        }
        return map;
    }


}

8.靜態頁面index.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>springboot整合redis之用戶手機驗證碼注冊登錄</title>
    <script type="text/javascript" src="js/jquery-1.9.1.js"></script>
</head>
<body>
      <!--  <h3>springboot整合redis之用戶手機驗證碼注冊登錄</h3>-->
      <div class="form-group has-feedback">
          <input type="tel" class="form-control" id="phone" placeholder="請輸入手機號" maxlength=11>
          <span class="glyphicon glyphicon-earphone form-control-feedback"></span>
      </div>
      <div class="row">
          <div class="col-xs-6 pull_left">
              <div class="form-group">
                  <input class="form-control" id="msg_num" placeholder="請輸入驗證碼">
              </div>
          </div>
          <div class="col-xs-6 pull_center">
              <div class="form-group">
                  <input type="button" class="btn btn-block btn-flat" id="verify_refresh" onclick="getMsgNum(this)" value="免費獲取驗證碼">
              </div>
          </div>
      </div>
      <div class="col-xs-12 pull_center">
          <button type="button" class="btn btn-block btn-flat" onclick="validateNum()">驗證</button>
      </div>


<!--js中的代碼-->
<script>

    var messageData;
    var wait = 120; // 短信驗證碼120秒后才可獲取下一個
    /**
     * 獲取驗證碼
     */
    function getMsgNum(that) {
        var phoneNumber = $('#phone').val();
        setButtonStatus(that); // 設置按鈕倒計時
        var obj = {
            phoneNumber: phoneNumber
        };
        $.ajax({
            url: 'sms/smsXxs', // 后台短信發送接口
            type: 'POST',
            dataType: 'json',
            contentType: "application/json",
            async: false, //false 同步
            data: JSON.stringify(obj),
            xhrFields: { //為true實現跨域訪問
                withCredentials: true
            },
            success: function (result) {
                if(result.isOk="OK") {
                    alert("驗證碼發送成功");
                    messageData = result;
                }else {
                    alert("驗證碼發送失敗")
                }
            },
            error: function (XMLHttpRequest, textStatus, errorThrown) {
                console.log(XMLHttpRequest.status);
                console.log(XMLHttpRequest.readyState);
                console.log(textStatus);
            }
        });
    }
    /**
     * 設置按鈕狀態
     */
    function setButtonStatus(that) {
        if (wait == 0) {
            that.removeAttribute("disabled");
            that.value="免費獲取驗證碼";
            wait = 60;
        } else {
            that.setAttribute("disabled", true);
            that.value=wait+"秒后可以重新發送";
            wait--;
            setTimeout(function() {
                setButtonStatus(that)
            }, 1000)
        }
    }
    /**
     * 注冊按鈕
     */
    function validateNum() {
        var data = {
            // verifyCode: messageData.verifyCode,
            // phone: messageData.phone,
            verifyCode: $('#msg_num').val(),
            phone: $('#phone').val(),
        };

        $.ajax({
            url: 'sms/validateNum', // 后台短信發送接口
            type: 'POST',
            dataType: 'json',
            contentType: "application/json",
            async: false, //false 同步
            data: JSON.stringify(data),
            xhrFields: { //為true實現跨域訪問
                withCredentials: true
            },
            success: function (result) {
                if(result.ruselt=="404") {
                    alert("過期驗證碼已失效");
                }else if(result.ruselt=="500"){
                    alert("驗證碼錯誤")
                }else{
                    alert("驗證成功");
                }
            },
            error: function (XMLHttpRequest, textStatus, errorThrown) {
                console.log(XMLHttpRequest.status);
                console.log(XMLHttpRequest.readyState);
                console.log(textStatus);
            }
        });

    }



</script>
</body>
</html>

完成以上代碼就可以運行項目了打開瀏覽器輸入http://localhost:8080/,就可訪問index.html文件,效果如下。。。。

輸入手機號碼點擊獲取驗證碼:

 

 

 

 隨后我們去的redis中查看是否將驗證碼保存到redis中,此時我們看見redis已經完成了緩存的實現並且設置了緩存超時時間。

輸入上手機短信驗證碼傳入進行一系列處理,后台獲取前台傳入的驗證碼同時獲取redis緩存中的驗證碼進行匹配該驗證碼是否有效、是否過期等問題驗證!

 總語:

在此個人只是做了一個小例子,各大網友可自行擴展功能業務,學習是永無止境的!


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM