Maison >base de données >Redis >Comment Redis met-il en œuvre l'envoi du code de vérification et limite le nombre d'envois quotidiens ?

Comment Redis met-il en œuvre l'envoi du code de vérification et limite le nombre d'envois quotidiens ?

WBOY
WBOYavant
2023-05-27 12:55:111898parcourir

1. Fonction

  • Entrez votre numéro de téléphone portable, cliquez sur Envoyer pour générer aléatoirement un code à six chiffres, valable 2 minutes

    #🎜 🎜##🎜 🎜#
  • Entrez le code de vérification, cliquez sur Vérifier, renvoyez le succès ou l'échec
  • Chaque numéro de téléphone mobile ne peut être saisi que 3 fois par jour
  • #🎜🎜 #

    2. Analyse

Chaque téléphone portable ne peut perdre que 3 fois par jour : augmenter +1 après à chaque envoi, lorsque la valeur est 3 L'invite ne peut pas être envoyée et l'heure d'expiration est la fin de la journée
  • Générez aléatoirement un code de vérification à 6 chiffres : RandomUtil (hutool)
  • Le code de vérification est valable 2 minutes : mettez-le dans redis et réglez le délai d'expiration sur 2 minutes
  • #🎜 🎜#

    Jugez si le code de vérification est cohérent : obtenez le code de vérification et saisissez-le depuis redis Comparez les codes de vérification

  • 3 Implémentez

    package cn.ken.blog.controller.common;
    
    import cn.hutool.core.date.DateUnit;
    import cn.hutool.core.date.DateUtil;
    import cn.hutool.core.util.RandomUtil;
    import cn.ken.blog.common.constant.Constants;
    import cn.ken.blog.common.domain.Result;
    import cn.ken.blog.common.enums.ErrorCodeEnum;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.data.redis.core.RedisTemplate;
    import org.springframework.scheduling.annotation.Scheduled;
    import org.springframework.util.ObjectUtils;
    import org.springframework.web.bind.annotation.GetMapping;
    import org.springframework.web.bind.annotation.RestController;
    
    import java.util.Date;
    import java.util.concurrent.TimeUnit;
    
    /**
     * 验证码控制器
     * @author Ken-Chy129
     * @date 2022/4/17 20:28
     */
    @RestController
    @SuppressWarnings(value = { "unchecked", "rawtypes" })
    public class CaptureController {
        
        @Autowired
        private RedisTemplate redisTemplate;
        
        // 生成验证码
        @GetMapping("getNumCode")
        public Result<String> getNumCode(String phone) {
            String captureLimitKey = Constants.CAPTCHA_LIMIT_KEY + phone;
            Integer counts = (Integer) redisTemplate.opsForValue().get(captureLimitKey);
            if (ObjectUtils.isEmpty(counts)) {
                // 今天第一次验证,故之前缓存中无该键
                // 距离今天结束剩下多少毫秒
                long expire = DateUtil.endOfDay(new Date()).between(new Date(), DateUnit.MS);
                redisTemplate.opsForValue().set(captureLimitKey, 1, expire, TimeUnit.MILLISECONDS);
            } else if (counts < 3) {
                // 没有超过限制次数
                redisTemplate.opsForValue().increment(captureLimitKey);
            } else {
                // 超过限制次数,不生成验证码,直接返回
                return new Result<String>().error(ErrorCodeEnum.OVER_LIMITS);
            }
            // 生成验证码
            String code = RandomUtil.randomNumbers(6); // 随机生成六位数
            String captureCodeKey = Constants.CAPTCHA_CODE_KEY + phone;
            redisTemplate.opsForValue().set(captureCodeKey, code, Constants.CAPTCHA_EXPIRATION, TimeUnit.MINUTES);
            return new Result<String>().success(captureCodeKey + ":" + code);
        }
        
        // 验证验证码
        @GetMapping("verify")
        public Result<String> verify(String phone, String code) {
            String captureCodeKey = Constants.CAPTCHA_CODE_KEY + phone;
            String realCode = (String) redisTemplate.opsForValue().get(captureCodeKey);
            if (ObjectUtils.isEmpty(realCode)) {
                // redis中不存在该用户生成的验证码,证明验证码以过期销毁
                return new Result<String>().error(ErrorCodeEnum.OVERDUE_CODE);
            }
            if (realCode.equals(code)) {
                return new Result<String>().success("验证成功");
            } else {
                return new Result<String>().error(ErrorCodeEnum.ERROR_CODE);
            }
        }
        
    //    @Scheduled(cron = "0 0 12 * * ?")
    //    private void clear() {
    //        redisTemplate.delete()
    //    }
    }
    // Constants类
    
    /**
     * 验证码 redis key
     */
    public static final String CAPTCHA_CODE_KEY = "captcha_codes:";
    
    /**
     * 每日限制 redis key
     */
    public static final String CAPTCHA_LIMIT_KEY = "captcha_limits:";
    /**
     * 验证码有效期(分钟)
     */
    public static final Integer CAPTCHA_EXPIRATION = 2;
    .

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
Cet article est reproduit dans:. en cas de violation, veuillez contacter admin@php.cn Supprimer