【畅购商城】校验用户名、手机号以及前置技术Redis和阿里大鱼短信验证码

    1. 搭建环境
      1. 后端web服务:changgou4-service-web
  1. 修改pom.xml文档

<?xml version="1.0" encoding="UTF-8"?>

<project xmlns="http://maven.apache.org/POM/4.0.0"

         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">

    <parent>

        <artifactId>changgou4-parent-ali</artifactId>

        <groupId>com.czxy.changgou</groupId>

        <version>1.0-SNAPSHOT</version>

    </parent>

    <modelVersion>4.0.0</modelVersion>

    <artifactId>changgou4_service_web</artifactId>

    <dependencies>

        <!--自定义项目-->

        <dependency>

            <groupId>com.czxy.changgou</groupId>

            <artifactId>changgou4_common_db</artifactId>

        </dependency>

        <dependency>

            <groupId>com.czxy.changgou</groupId>

            <artifactId>changgou4_pojo</artifactId>

        </dependency>

        <!--web起步依赖-->

        <dependency>

            <groupId>org.springframework.boot</groupId>

            <artifactId>spring-boot-starter-web</artifactId>

        </dependency>

        <!-- nacos 客户端 -->

        <dependency>

            <groupId>com.alibaba.nacos</groupId>

            <artifactId>nacos-client</artifactId>

        </dependency>

        <!-- nacos 服务发现 -->

        <dependency>

            <groupId>com.alibaba.cloud</groupId>

            <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>

        </dependency>

        <!--redis-->

        <dependency>

            <groupId>org.springframework.boot</groupId>

            <artifactId>spring-boot-starter-data-redis</artifactId>

        </dependency>

        <dependency>

            <groupId>redis.clients</groupId>

            <artifactId>jedis</artifactId>

        </dependency>

        <!--swagger2-->

        <dependency>

            <groupId>io.springfox</groupId>

            <artifactId>springfox-swagger2</artifactId>

        </dependency>

        <dependency>

            <groupId>io.springfox</groupId>

            <artifactId>springfox-swagger-ui</artifactId>

        </dependency>

    </dependencies>

</project>

  1. 创建application.yml文档

#端口号

server:

  port: 8081

spring:

  application:

    name: web-service          #服务名

  datasource:

    driverClassName: com.mysql.jdbc.Driver

    url: jdbc:mysql://127.0.0.1:3306/changgou_db?useUnicode=true&characterEncoding=utf8

    username: root

    password: 1234

    druid:    #druid 连接池配置

      initial-size: 1       #初始化连接池大小

      min-idle: 1           #最小连接数

      max-active: 20        #最大连接数

      test-on-borrow: true  #获取连接时候验证,会影响性能

  redis:

    database:   0

    host: 127.0.0.1

    port: 6379

  cloud:

    nacos:

      discovery:

        server-addr: 127.0.0.1:8848   #nacos服务地址

    sentinel:

      transport:

        dashboard: 127.0.0.1:8080

  1. 创建启动类

package com.czxy.changgou4;

import org.springframework.boot.SpringApplication;

import org.springframework.boot.autoconfigure.SpringBootApplication;

import org.springframework.cloud.client.discovery.EnableDiscoveryClient;

/**

 * @author 桐叔

 * @email liangtong@itcast.cn

 */

@SpringBootApplication

@EnableDiscoveryClient

public class Web-serviceApplication {

    public static void main(String[] args) {

        SpringApplication.run( Web-serviceApplication.class , args );

    }

}

      1. 后端创建JavaBean:User
  1. 在changgou4-pojo项目中添加User对象

package com.czxy.changgou4.pojo;

import com.baomidou.mybatisplus.annotation.IdType;

import com.baomidou.mybatisplus.annotation.TableField;

import com.baomidou.mybatisplus.annotation.TableId;

import com.baomidou.mybatisplus.annotation.TableName;

import lombok.AllArgsConstructor;

import lombok.Data;

import lombok.NoArgsConstructor;

import lombok.ToString;

import java.beans.Transient;

import java.util.Date;

/**  与数据库对应JavaBean

 * Created by liangtong.

 */

@TableName("tb_user")

@Data

@NoArgsConstructor

@AllArgsConstructor

public class User {

    /*

        CREATE TABLE `tb_user` (

           `id` int(10) unsigned NOT NULL AUTO_INCREMENT,

           `created_at` timestamp NULL DEFAULT NULL,

           `updated_at` timestamp NULL DEFAULT NULL,

           `email` varchar(255) COLLATE utf8_unicode_ci DEFAULT NULL COMMENT 'Email',

           `mobile` varchar(20) COLLATE utf8_unicode_ci NOT NULL COMMENT '手机号码',

           `username` varchar(255) COLLATE utf8_unicode_ci NOT NULL COMMENT '昵称',

           `password` char(60) COLLATE utf8_unicode_ci NOT NULL COMMENT '密码',

           `face` varchar(255) COLLATE utf8_unicode_ci DEFAULT NULL COMMENT '头像',

           `expriece` int(10) unsigned DEFAULT '0' COMMENT '经验值',

           PRIMARY KEY (`id`),

           UNIQUE KEY `users_mobile_unique` (`mobile`),

           UNIQUE KEY `users_name_unique` (`name`),

           UNIQUE KEY `users_email_unique` (`email`)

         ) ENGINE=InnoDB AUTO_INCREMENT=27 DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci

     */

    @TableId(value="id",type = IdType.AUTO)

    private Long id;

    @TableField(value="username")

    private String username;

    @TableField(value="password")

    private String password;

    @TableField(value="face")

    private String face;

    @TableField(value="expriece")

    private Integer expriece;

    @TableField(value="email")

    private String email;

    @TableField(value="mobile")

    private String mobile;

    @TableField(value="created_at")

    private Date createdAt;

    @TableField(value="updated_at")

    private Date updatedAt;

    @TableField(exist = false)

    private String code;

    @TableField(exist = false)

    private String password_confirm;

}

      1. 前端页面:创建公共组件
  1. 1)删除components目录下所有内容,并创建3个新组件

  1. 2)创建 TopNav.vue组件,用于配置“顶部导航”

<template>

  <!-- 顶部导航 start -->

  <div class="topnav">

    <div class="topnav_bd w990 bc">

      <div class="topnav_left">

      </div>

      <div class="topnav_right fr">

        <ul>

          <li>您好,欢迎来到畅购![<a href="login.html">登录</a>] [<a href="register.html">免费注册</a>] </li>

          <li class="line">|</li>

          <li>我的订单</li>

          <li class="line">|</li>

          <li>客户服务</li>

        </ul>

      </div>

    </div>

  </div>

  <!-- 顶部导航 end -->

</template>

<script>

export default {

}

</script>

<style>

</style>

  1. 3)创建 HeaderLogo.vue组件,用于配置“页面头部,仅有LOGO”

<template>

  <!-- 页面头部 start,内容太多,省略了中间过程 -->

  <div class="header w990 bc mt15">

    <div class="logo w990">

      <h2 class="fl"><a href="index.html"><img src="/images/logo.png" alt="畅购商城"></a></h2>

    </div>

  </div>

  <!-- 页面头部 end -->

</template>

<script>

export default {

}

</script>

<style>

</style>

  1. 4)创建 Footer.vue组件,用于配置“底部版权”

<template>

  <!-- 底部版权 start -->

  <div class="footer w1210 bc mt15">

    <p class="links">

      <a href="">关于我们</a> |

      <a href="">联系我们</a> |

      <a href="">人才招聘</a> |

      <a href="">商家入驻</a> |

      <a href="">千寻网</a> |

      <a href="">奢侈品网</a> |

      <a href="">广告服务</a> |

      <a href="">移动终端</a> |

      <a href="">友情链接</a> |

      <a href="">销售联盟</a> |

      <a href="">畅购论坛</a>

    </p>

    <p class="copyright">

       © 2006-2020 畅购网上商城 版权所有,并保留所有权利。  ICP备案证书号:京ICP证070359号

    </p>

    <p class="auth">

      <a href=""><img src="/images/xin.png" alt="" /></a>

      <a href=""><img src="/images/kexin.jpg" alt="" /></a>

      <a href=""><img src="/images/police.jpg" alt="" /></a>

      <a href=""><img src="/images/beian.gif" alt="" /></a>

    </p>

  </div>

  <!-- 底部版权 end -->

</template>

<script>

export default {

}

</script>

<style>

</style>

    1. 用户注册:用户名占用

      1. 接口

http://localhost:10010/web-service/user/checkusername

{

"username":"jack1"

}

      1. 后端
  1. 创建三层需要的接口或类

  1. 步骤一:创建UserMapper,编写findByUsername()完成“通过用户名查询用户”

package com.czxy.changgou4.mapper;

import com.baomidou.mybatisplus.core.mapper.BaseMapper;

import com.czxy.changgou4.pojo.User;

import org.apache.ibatis.annotations.Param;

import org.apache.ibatis.annotations.Select;

/**

 * Created by liangtong.

 */

@org.apache.ibatis.annotations.Mapper

public interface UserMapper extends BaseMapper<User> {

    /**

     * 通过用户名查询

     * @param username

     * @return

     */

    @Select("select * from tb_user where username = #{username}")

    User findByUsername(@Param("username") String username);

}

  1. 步骤二:创建UserService接口,查询功能

package com.czxy.changgou4.service;

import com.baomidou.mybatisplus.extension.service.IService;

import com.czxy.changgou4.pojo.User;

/**

 * @author 桐叔

 * @email liangtong@itcast.cn

 */

public interface UserService extends IService<User> {

    /**

     * 通过用户名查询

     * @param username

     * @return

     */

    public User findByUsername(String username);

}

  1. 步骤三:创建UserServiceImpl实现类,查询功能

package com.czxy.changgou4.service.impl;

import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;

import com.czxy.changgou4.mapper.UserMapper;

import com.czxy.changgou4.pojo.User;

import com.czxy.changgou4.service.UserService;

import org.springframework.stereotype.Service;

import org.springframework.transaction.annotation.Transactional;

/**

 * @author 桐叔

 * @email liangtong@itcast.cn

 */

@Service

@Transactional

public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements UserService {

    @Override

    public User findByUsername(String username) {

        return baseMapper.findByUsername(username);

    }

}

  1. 步骤四:创建UserController,完成用户名检查

package com.czxy.changgou4.controller;

import com.czxy.changgou4.pojo.User;

import com.czxy.changgou4.service.UserService;

import com.czxy.changgou4.vo.BaseResult;

import org.springframework.web.bind.annotation.*;

import javax.annotation.Resource;

/**

 * Created by liangtong.

 */

@RestController

@RequestMapping("/user")

public class UserController {

    @Resource

    private UserService userService;

    @PostMapping("/checkusername")

    public BaseResult checkUsername(@RequestBody User user){

        //查询用户

        User findUser = userService.findByUsername( user.getUsername() );

        //判断

        if(findUser != null){

            return BaseResult.error("用户名已经存在");

        } else {

            return BaseResult.ok("用户名可用");

        }

    }

}

      1. 前端
  1. 步骤一:创建Register.vue

  1. 步骤二:添加公共组件

<template>

  <div>

    <TopNav></TopNav>

    <div style="clear:both;"></div>

    <HeaderLogo></HeaderLogo>

    <div style="clear:both;"></div>

    <!-- 正文 -->

    <div style="clear:both;"></div>

    <Footer></Footer>

  </div>

</template>

<script>

import TopNav from '../components/TopNav'

import HeaderLogo from '../components/HeaderLogo'

import Footer from '../components/Footer'

export default {

  head: {

    title: '用户注册'

  },

  components : {

    TopNav,

    HeaderLogo,

    Footer

  }

}

</script>

<style>

</style>

  1. 步骤三:编写注册表单,并导入独有样式

<template>

  <div>

    <TopNav></TopNav>

    <div style="clear:both;"></div>

    <HeaderLogo></HeaderLogo>

    <div style="clear:both;"></div>

    <!-- 正文 -->

      <!-- 登录主体部分start -->

  <div class="login w990 bc mt10 regist">

    <div class="login_hd">

      <h2>用户注册</h2>

      <b></b>

    </div>

    <div class="login_bd">

      <div class="login_form fl">

        <form action="" method="post">

          <ul>

            <li>

              <label for="">用户名:</label>

              <input type="text" class="txt" name="username" />

              <p>3-20位字符,可由中文、字母、数字和下划线组成</p>

              <p class="error">用户名已存在</p>

            </li>

            <li>

              <label for="">密码:</label>

              <input type="password" class="txt" name="password" />

              <p>6-20位字符,可使用字母、数字和符号的组合,不建议使用纯数字、纯字母、纯符号</p>

            </li>

            <li>

              <label for="">确认密码:</label>

              <input type="password" class="txt" name="password" />

              <p>请再次输入密码</p>

            </li>

            <li>

              <label for="">手机号码:</label>

              <input type="text" class="txt"  name="mobile" />

              <p>请输入手机号码</p>

            </li>

            <li class="checkcode">

              <label for="">验证码:</label>

              <input type="text"  name="checkcode" />

              <button >

                发送验证码<span>5秒</span>

              </button>

            </li>

            <li>

              <label for=""> </label>

              <input type="checkbox" class="chb" checked="checked" /> 我已阅读并同意《用户注册协议》

            </li>

            <li>

              <label for=""> </label>

              <input type="submit" value="" class="login_btn" />

            </li>

          </ul>

        </form>

      </div>

      <div class="mobile fl">

        <h3>手机快速注册</h3>

        <p>中国大陆手机用户,编辑短信 “<strong>XX</strong>”发送到:</p>

        <p><strong>1069099988</strong></p>

      </div>

    </div>

  </div>

  <!-- 登录主体部分end -->

    <div style="clear:both;"></div>

    <Footer></Footer>

  </div>

</template>

<script>

import TopNav from '../components/TopNav'

import HeaderLogo from '../components/HeaderLogo'

import Footer from '../components/Footer'

export default {

  head: {

    title: '用户注册',

    link: [

      {rel:'stylesheet',href:'style/login.css'}

    ],

    script: [

      { type: 'text/javascript', src: 'js/header.js' },

      { type: 'text/javascript', src: 'js/index.js' },

    ]

  },

  components : {

    TopNav,

    HeaderLogo,

    Footer

  }

}

</script>

<style>

</style>

  1. 步骤四:修改api.js,编写检查用户名的ajax函数

const request = {

  test : ()=> {

    return axios.get('/test')

  },

  //检查用户名

  checkUsername : ( username )=> {

    return axios.post('/web-service/user/checkusername', { username })

  }

}

  1. 步骤五:修改Register.vue页面,完成检查功能
    1. 发送ajax进行用户名是否可用检查
    2. 如果可用,显示对应信息,并使用success样式显示
    3. 如果不可用,显示对应信息,并使用error样式提示

<template>

  <div>

    <TopNav></TopNav>

    <div style="clear:both;"></div>

    <HeaderLogo></HeaderLogo>

    <div style="clear:both;"></div>

    <!-- 正文 -->

      <!-- 登录主体部分start -->

  <div class="login w990 bc mt10 regist">

    <div class="login_hd">

      <h2>用户注册</h2>

      <b></b>

    </div>

    <div class="login_bd">

      <div class="login_form fl">

        <form action="" method="post">

          <ul>

            <li>

              <label for="">用户名:</label>

              <input type="text" class="txt" name="username" v-model="user.username" @blur="checkUsernameFn" />

              <p>3-20位字符,可由中文、字母、数字和下划线组成</p>

              <p :class="userMsg.usernameData.code == 20000 ? 'success' : 'error'">{{userMsg.usernameData.message}} </p>

            </li>

            <li>

              <label for="">密码:</label>

              <input type="password" class="txt" name="password" />

              <p>6-20位字符,可使用字母、数字和符号的组合,不建议使用纯数字、纯字母、纯符号</p>

            </li>

            <li>

              <label for="">确认密码:</label>

              <input type="password" class="txt" name="password" />

              <p>请再次输入密码</p>

            </li>

            <li>

              <label for="">手机号码:</label>

              <input type="text" class="txt"  name="mobile" />

              <p>请输入手机号码</p>

            </li>

            <li class="checkcode">

              <label for="">验证码:</label>

              <input type="text"  name="checkcode" />

              <button >

                发送验证码<span>5秒</span>

              </button>

            </li>

            <li>

              <label for=""> </label>

              <input type="checkbox" class="chb" checked="checked" /> 我已阅读并同意《用户注册协议》

            </li>

            <li>

              <label for=""> </label>

              <input type="submit" value="" class="login_btn" />

            </li>

          </ul>

        </form>

      </div>

      <div class="mobile fl">

        <h3>手机快速注册</h3>

        <p>中国大陆手机用户,编辑短信 “<strong>XX</strong>”发送到:</p>

        <p><strong>1069099988</strong></p>

      </div>

    </div>

  </div>

  <!-- 登录主体部分end -->

    <div style="clear:both;"></div>

    <Footer></Footer>

  </div>

</template>

<script>

import TopNav from '../components/TopNav'

import HeaderLogo from '../components/HeaderLogo'

import Footer from '../components/Footer'

export default {

  head: {

    title: '用户注册',

    link: [

      {rel:'stylesheet',href:'style/login.css'}

    ],

    script: [

      { type: 'text/javascript', src: 'js/header.js' },

      { type: 'text/javascript', src: 'js/index.js' },

    ]

  },

  components : {

    TopNav,

    HeaderLogo,

    Footer

  },

  data() {

    return {

      user : { //表单绑定对象

        username : ""

      },

      userMsg : { //错误信息

        usernameData : ""

      }

    }

  },

  methods: {

    async checkUsernameFn() {

      let {data} = await this.$request.checkUsername( this.user.username )

      this.userMsg.usernameData = data

    }

  },

}

</script>

<style>

</style>

    1. 用户注册:手机号检查
      1. 接口

http://localhost:10010/web-service/user/checkmobile

{

"mobile":"13344445555"

}

      1. 后端

  1. 步骤一:修改UserService,添加 findByMobile() 方法,进行电话号码的查询

/**

 * 通过手机号查询

 * @param mobile

 * @return

 */

User findByMobile(String mobile);

  1. 步骤二:编写UserServiceImpl,实现findByMobile() 方法

@Override

public User findByMobile(String mobile) {

    // 拼凑条件

    QueryWrapper queryWrapper = new QueryWrapper();

    queryWrapper.eq("mobile", mobile);

    // 查询一个

    List<User> list = baseMapper.selectList(queryWrapper);

    if(list.size() == 1) {

        return list.get(0);

    }

    return null;

}

  1. 步骤三:修改UserController,添加checkMobile() 方法

/**

 * 通过手机号查询

 * @param user

 * @return

 */

@PostMapping("/checkmobile")

public BaseResult checkMobile(@RequestBody User user){

    //查询用户

    User findUser = userService.findByMobile( user.getMobile() );

    //判断

    if(findUser != null){

        return BaseResult.error("电话号码已经注册");

    } else {

        return BaseResult.ok("电话号码可用");

    }

}

      1. 前端
  1. 步骤一:修改api.js,添加 checkMobile() 函数

const request = {

  test : ()=> {

    return axios.get('/test')

  },

  //检查用户名

  checkUsername : ( username )=> {

    return axios.post('/web-service/user/checkusername', { username })

  },

  //检查电话号码

  checkMobile : ( mobile )=> {

    return axios.post('/web-service/user/checkmobile', { mobile })

  }

}

  1. 步骤二:修改Register.vue,添加 checkMobileFn() 进行手机号检查

  methods: {

    async checkUsernameFn() {

      //检查用户名

      let {data} = await this.$request.checkUsername( this.user.username )

      this.userMsg.usernameData = data

    },

    async checkMobileFn() {

      //检查电话

      let {data} = await this.$request.checkMobile( this.user.mobile )

      this.userMsg.mobileData = data

    }

  },

  1. 步骤三:编写需要的2个变量

  data() {

    return {

      user : {  //表单封装数据

        username : "",

        mobile : ""

      },

      userMsg : { //错误提示数据

        usernameData : "",

        mobileData : ""

      }

    }

  },

  1. 步骤四:处理页面

<li>

              <label for="">手机号码:</label>

              <input type="text" class="txt"  name="mobile"  v-model="user.mobile" @blur="checkMobileFn" />

              <p>请输入手机号码</p>

              <p :class="userMsg.mobileData.code == 1 ? 'success' : 'error'">{{userMsg.mobileData.message}} </p>

            </li>

  1. 完整版

<template>

  <div>

    <TopNav></TopNav>

    <div style="clear:both;"></div>

    <HeaderLogo></HeaderLogo>

    <div style="clear:both;"></div>

    <!-- 正文 -->

      <!-- 登录主体部分start -->

  <div class="login w990 bc mt10 regist">

    <div class="login_hd">

      <h2>用户注册</h2>

      <b></b>

    </div>

    <div class="login_bd">

      <div class="login_form fl">

        <form action="" method="post">

          <ul>

            <li>

              <label for="">用户名:</label>

              <input type="text" class="txt" name="username" v-model="user.username" @blur="checkUsernameFn" />

              <p>3-20位字符,可由中文、字母、数字和下划线组成</p>

              <p :class="userMsg.usernameData.code == 1 ? 'success' : 'error'">{{userMsg.usernameData.message}} </p>

            </li>

            <li>

              <label for="">密码:</label>

              <input type="password" class="txt" name="password" />

              <p>6-20位字符,可使用字母、数字和符号的组合,不建议使用纯数字、纯字母、纯符号</p>

            </li>

            <li>

              <label for="">确认密码:</label>

              <input type="password" class="txt" name="password" />

              <p>请再次输入密码</p>

            </li>

            <li>

              <label for="">手机号码:</label>

              <input type="text" class="txt"  name="mobile"  v-model="user.mobile" @blur="checkMobileFn" />

              <p>请输入手机号码</p>

              <p :class="userMsg.mobileData.code == 1 ? 'success' : 'error'">{{userMsg.mobileData.message}} </p>

            </li>

            <li class="checkcode">

              <label for="">验证码:</label>

              <input type="text"  name="checkcode" />

              <button >

                发送验证码<span>5秒</span>

              </button>

            </li>

            <li>

              <label for=""> </label>

              <input type="checkbox" class="chb" checked="checked" /> 我已阅读并同意《用户注册协议》

            </li>

            <li>

              <label for=""> </label>

              <input type="submit" value="" class="login_btn" />

            </li>

          </ul>

        </form>

      </div>

      <div class="mobile fl">

        <h3>手机快速注册</h3>

        <p>中国大陆手机用户,编辑短信 “<strong>XX</strong>”发送到:</p>

        <p><strong>1069099988</strong></p>

      </div>

    </div>

  </div>

  <!-- 登录主体部分end -->

    <div style="clear:both;"></div>

    <Footer></Footer>

  </div>

</template>

<script>

import TopNav from '../components/TopNav'

import HeaderLogo from '../components/HeaderLogo'

import Footer from '../components/Footer'

export default {

  head: {

    title: '用户注册',

    link: [

      {rel:'stylesheet',href:'style/login.css'}

    ],

    script: [

      { type: 'text/javascript', src: 'js/header.js' },

      { type: 'text/javascript', src: 'js/index.js' },

    ]

  },

  components : {

    TopNav,

    HeaderLogo,

    Footer

  },

  data() {

    return {

      user : {  //表单封装数据

        username : "",

        mobile : "",

        password : "",

        code : ""

      },

      userMsg : { //错误提示数据

        usernameData : "",

        mobileData : ""

      }

    }

  },

  methods: {

    async checkUsernameFn() {

      //检查用户名

      let {data} = await this.$request.checkUsername( this.user.username )

      this.userMsg.usernameData = data

    },

    async checkMobileFn() {

      //检查电话

      let {data} = await this.$request.checkMobile( this.user.mobile )

      this.userMsg.mobileData = data

    }

  },

}

</script>

<style>

</style>

    1. 用户注册:前置技术--Redis
      1. 安装redis
  1. linux版
  2. window

      1. 可访问

      1. 图形化工具

      1. 基本使用
  1. yml配置

spring:

  application:

    name: test-service          #服务名

  redis:

    database:   0

    host: 127.0.0.1

    port: 6379

  1. 配置类

package com.czxy.config;

import org.springframework.context.annotation.Bean;

import org.springframework.context.annotation.Configuration;

import org.springframework.data.redis.connection.RedisConnectionFactory;

import org.springframework.data.redis.core.StringRedisTemplate;

/**

 * @author 桐叔

 * @email liangtong@itcast.cn

 */

@Configuration

public class RedisConfiguration {

    @Bean

    public StringRedisTemplate stringRedisTemplate(RedisConnectionFactory redisConnectionFactory) {

        return new StringRedisTemplate(redisConnectionFactory);

    }

}

  1. 测试

package com.czxy;

import org.junit.Test;

import org.junit.runner.RunWith;

import org.springframework.boot.test.context.SpringBootTest;

import org.springframework.data.redis.core.StringRedisTemplate;

import org.springframework.test.context.junit4.SpringRunner;

import javax.annotation.Resource;

/**

 * @author 桐叔

 * @email liangtong@itcast.cn

 */

@RunWith(SpringRunner.class)

@SpringBootTest(classes = TestApplication.class)

public class RedisTest {

    @Resource

    private StringRedisTemplate stringRedisTemplate;

    @Test

    public void testDemo() {

        stringRedisTemplate.opsForValue().set("username","下课");

    }

}

    1. 用户注册:前置技术--阿里大鱼

    1. 用户注册:短信验证码
      1. 分析

      1. 接口

POST http://localhost:10010/web-service/sms

{

"mobile":"13344445555",

"username": "jack"

}

      1. 后端
  1. 创建 SmsController类,调用阿里大鱼工具类,发送短信。

package com.czxy.changgou4.controller;

import com.aliyuncs.dysmsapi.model.v20170525.SendSmsResponse;

import com.czxy.changgou4.pojo.User;

import com.czxy.changgou4.utils.SmsUtil;

import com.czxy.changgou4.vo.BaseResult;

import org.apache.commons.lang.RandomStringUtils;

import org.springframework.data.redis.core.StringRedisTemplate;

import org.springframework.http.ResponseEntity;

import org.springframework.web.bind.annotation.PostMapping;

import org.springframework.web.bind.annotation.RequestBody;

import org.springframework.web.bind.annotation.RequestMapping;

import org.springframework.web.bind.annotation.RestController;

import javax.annotation.Resource;

import java.util.concurrent.TimeUnit;

/**

 * Created by liangtong.

 */

@RestController

@RequestMapping("/sms")

public class SmsController {

    @Resource

    private StringRedisTemplate redisTemplate;

    @PostMapping

    public BaseResult sendSms(@RequestBody User user){

        long start = System.currentTimeMillis();

        try {

            //发送短信

            //1 生产验证码

            String code = RandomStringUtils.randomNumeric(4);

            System.out.println("验证码:" + code);

            //2 并存放到reids , key: "sms_register" + 手机号 , value:验证码 , 1小时

            redisTemplate.opsForValue().set( "sms_register" + user.getMobile() , code , 1 , TimeUnit.HOURS);

            /**/

            //3 发送短信

            SendSmsResponse smsResponse = SmsUtil.sendSms(user.getMobile(), user.getUsername() , code, "", "");

            //https://help.aliyun.com/document_detail/55284.html?spm=5176.doc55322.6.557.KvvIJx

            if("OK".equalsIgnoreCase(smsResponse.getCode())){

                return BaseResult.ok("发送成功");

            } else {

                return BaseResult.error(smsResponse.getMessage());

            }

            /*

            //模拟数据

            System.out.println("验证码:" + code);

            return BaseResult.ok("发送成功");

            */

        } catch (Exception e) {

            long end = System.currentTimeMillis();

            System.out.println( end - start);

            return BaseResult.error("发送失败" );

        }

    }

}

      1. 前端
  1. 步骤一:修改apiclient.js,发送短信ajax操作

  //发短信

  sendSms : ( user )=> {

    return axios.post('/web-service/sms', user )

  }

  1. 步骤二:修改Register.vue页面,给“发送验证码”绑定点击事件 sendSmsFn

<button  @click.prevent="sendSmsFn" >

                发送验证码<span>5秒</span>

              </button>

  1. 步骤三:修改Register.vue页面,编写sendSmsFn函数,建议采用  ajax..then()..catch 可以处理异常

sendSmsFn () {

      this.$request.sendSms( this.user )

      .then(( response )=>{

        //发送短信的提示信息

        this.userMsg.smsData = response.data

      })

      .catch(( error )=>{

        //错误提示信息

        alert( error.message )

      })

    }

  1. 步骤四:修改Register.vue页面,提供变量smsData

userMsg : { //错误提示数据

        usernameData : "",

        mobileData : "",

        smsData : ""

      }

  1. 步骤五:修改Register.vue页面,显示 smsData提示信息

<p :class="userMsg.smsData.code == 1 ? 'success' : 'error'">{{userMsg.smsData.message}} </p>

      1. 倒计时
  1. 步骤一:提供3个变量,用于控制倒计时

      btnDisabled : false,  //倒计时控制变量

     seconds : 5,           //默认倒计时秒数

      timer : null,         //接收定时器,清除定时器

  1. 步骤二:在标签上面控制倒计时的显示

<button :disabled="btnDisabled" @click.prevent="sendSmsFn" >

      发送验证码<span v-show="btnDisabled">{{seconds}}</span>

</button>

  1. 步骤三:发送短信后,开启倒计时控制

sendSmsFn () {

      this.$request.sendSms( this.user )

      .then(( response )=>{

        //发送短信的提示信息

        this.userMsg.smsData = response.data

        //按钮不可用

        this.btnDisabled = true;

          //倒计时

          this.timer = setInterval( ()=>{

            if(this.seconds <= 1){

              //结束

              // 重置秒数

              this.seconds = 5;

              // 按钮可用

              this.btnDisabled = false;

              // 停止定时器

              clearInterval(this.timer);

            } else {

              this.seconds --;

            }

          } , 1000);

      })

      .catch(( error )=>{

        //错误提示信息

        alert( error.message )

      })

    }

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/497714.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[创业之路-222]:波士顿矩阵与GE矩阵在业务组合选中作用、优缺点比较

目录 一、波士顿矩阵 1、基本原理 2、各象限产品的定义及战略对策 3、应用 4、优点与局限性 二、技术成熟度模型与产品生命周期模型的配对 1、技术成熟度模型 2、产品生命周期模型 3、技术成熟度模型与产品生命周期模型的配对 三、产品生命周期与产品类型的对应关系 …

第三方接口设计注意要点

实际工作中&#xff0c;我们会遇到与三方系统对接的情形&#xff0c;比如对接短信服务、支付服务、地图服务、以及一些外部业务系统的调用和回调等等&#xff0c;不论是我们调用第三方接口还是我们为其他系统提供接口服务&#xff0c;调用过程中会遇到一些大大小小的问题和吐槽…

折腾日记:如何让吃灰笔记本发挥余热——搭建一个相册服务

背景 之前写过&#xff0c;我在家里用了一台旧的工作站笔记本做了服务器&#xff0c;连上一个绿联的5位硬盘盒实现简单的网盘功能&#xff0c;然而&#xff0c;还是觉的不太理想&#xff0c;比如使用filebrowser虽然可以备份文件和图片&#xff0c;当使用手机使用网页&#xf…

【设计与实现】基于Bootstrap的地方旅游管理系统的设计与实现

目录 第一章 绪论 1.1 研究现状 1.2 设计原则 1.3 研究内容 第四章 系统设计 4.1系统结构设计 4.2系统顺序图设计 4.3数据库设计 第五章 系统实现 5.1登录模块的实现 第一章 绪论 1.1 研究现状 时代的发展&#xff0c;我们迎来了数字化信息时代&#xff0c;它正在渐…

人工智能与区块链的碰撞:双剑合璧的创新前景

引言 人工智能&#xff08;AI&#xff09;与区块链技术&#xff0c;这两项曾经各自独立发展的前沿科技&#xff0c;如今正逐步走向融合。人工智能通过强大的数据处理能力和智能决策能力&#xff0c;在各个领域掀起了革命性的变革&#xff1b;而区块链凭借其去中心化、不可篡改的…

HarmonyOS NEXT 实战之元服务:静态案例效果---我的热门应用服务

背景&#xff1a; 前几篇学习了元服务&#xff0c;后面几期就让我们开发简单的元服务吧&#xff0c;里面丰富的内容大家自己加&#xff0c;本期案例 仅供参考 先上本期效果图 &#xff0c;里面图片自行替换 效果图1完整代码案例如下&#xff1a; Index import { authentica…

ArcGIS Pro地形图四至角图经纬度标注与格网标注

今天来看看ArcGIS Pro 如何在地形图上设置四至角点的经纬度。方里网标注。如下图的地形图左下角经纬度标注。 如下图方里网的标注 如下为本期要介绍的例图&#xff0c;如下&#xff1a; 图片可点击放大 接下来我们来介绍一下 推荐学习&#xff1a;GIS入门模型构建器Arcpy批量…

数字图像处理

一 形态学处理 ①二值图像 PS&#xff1a;1&#xff08;255&#xff09;代表的是白 0代表的是黑&#xff08;0就是什么都看不见&#xff0c;就是黑&#xff09; ②灰度图像 ③彩色图像 ④数学形态学基础&#xff1a;是分析几何形状和结构的数学方法&#xff0c;它建立在…

linux-软硬链接

我们今天再来聊一下这个"软硬链接"的问题. 目录 1. 软硬链接长什么样?2. 软连接和硬链接的特征 和 应用2.1 软连接特征 及其 应用?①软连接是什么?②软连接的应用1: 快捷方式③软连接的应用2: 方便维护库文件 2.2 硬连接特征 及其 应用?①硬链接是什么?②引用计…

SpringCloud 系列教程:微服务的未来(三)IService接口的业务实现

本文将介绍 IService 接口的基本业务操作、复杂业务操作、Lambda 方法的使用以及批量增加操作&#xff0c;帮助开发者深入了解如何高效地利用 MyBatis-Plus 提供的功能进行数据库操作。无论是简单的单表查询&#xff0c;还是复杂的多表联动&#xff0c;甚至是大数据量的批量操作…

Linux第100步_Linux之设置LCD作为终端控制台和LCD背光调节

KMS是Kemmel Mode Setting的缩写&#xff0c;内核显示模式设置。它主要负责显示的控制&#xff0c;包括屏幕分辨率、屏幕刷新率和颜色深度等等。 CRTC是指显示控制器&#xff0c;在DRM里有多个显存&#xff0c;通过操作CRTC来控制要显示那个显存。 KMS包含了FB框架。DRM驱动默…

解决pycharm无法识别miniconda

解决pycharm无法识别miniconda 选中 conda.bat 点击 Load Enviroments

云手机群控能用来做什么?

随着云手机的发展&#xff0c;云手机群控技术逐渐从小众的游戏多开工具&#xff0c;发展为涵盖多个领域的智能操作平台。不论是手游搬砖、短视频运营&#xff0c;还是账号养成等场景&#xff0c;云手机群控都展现出了强大的应用潜力。本文将为大家详细解析云手机群控的应用场景…

道路倒角 三角网 两侧偏移

public void 多段线和直线两侧缓冲区(){List<Curve> ents1 Z.db.SelectEntities<Curve>();List<Polyline> ents Z.db.CurvesToPolyLines2(ents1);//Z.db.SelectEntities<Polyline>();double offsetDistance 5.0;//p距离double offsetDistance2 1.0…

patch补丁制作,合入,卸载的方法

创建PATCH目录&#xff0c;进入该目录下&#xff0c;创建文件夹old, new, 创建文件1.c&#xff1b; 1.c内容如下&#xff1a; 在new下修改1.c&#xff1a; 开始制作1.patch diff -Naur ./old/1.c ./new/1.c > 1.patch 进入 vi 1.patch&#xff1a; 1.patch内容如下&#…

【基础篇】一、MySQL数据库基础知识

文章目录 Ⅰ. 什么是数据库1、普通文件的缺点2、数据库的概念3、主流数据库4、MySQL Ⅱ. MySQL中客户端、服务端、数据库的关系Ⅲ. 见一见数据库1、数据库文件存放的位置2、创建数据库3、使用数据库4、创建数据库表结构5、表中插入数据6、查询表中数据7、数据的存储逻辑 &#…

电脑vcruntime140.dll丢失的解决方法!vcruntime140.dll丢失是

一、文件丢失问题&#xff1a;vcruntime140.dll丢失的解决方法 vcruntime140.dll是Visual C Redistributable for Visual Studio的一个关键组件&#xff0c;许多应用程序和游戏都需要它才能正常运行。当系统提示vcruntime140.dll丢失时&#xff0c;通常意味着你的系统中缺少了…

【SpringBoot教程】IDEA快速搭建正确的SpringBoot版本和Java版本的项目

&#x1f64b;大家好&#xff01;我是毛毛张! &#x1f308;个人首页&#xff1a; 神马都会亿点点的毛毛张 &#x1f349;今天毛毛张分享的是在做SpringBoot项目中遇到的三个问题&#xff0c;这三个问题看似都是小问题&#x1f353;&#xff0c;但是却是做项目的基础常识⚠️…

MultiDiff 论文解读

一、CameraCtrl AnimateDiff->MotionCtrl->CameraCtrl CameraCtrl将多帧图像的Plucker射线输入到Camera Encoder&#xff0c;Plucker射线可以表示每个像素对应的光线方向。 Camera Encoder包括ResNet block和Temporal Attention&#xff0c;来提取每一帧相机位姿的时序…

C语言性能优化:从基础到高级的全面指南

引言 C 语言以其高效、灵活和功能强大而著称&#xff0c;被广泛应用于系统编程、嵌入式开发、游戏开发等领域。然而&#xff0c;要写出高性能的 C 语言代码&#xff0c;需要对 C 语言的特性和底层硬件有深入的了解。本文将详细介绍 C 语言性能优化的背后技术&#xff0c;并通过…