一.技术及思路
二.案例编写
-
配置环境:使用docker新建redis,mysql,rabbitmq的容器
- MySQL:
- 启动docker服务
-
systemctl start docker
- 查看docker服务状态
-
systemctl status docker
- 查看目录
- 进入mysql目录下,然后新创建一个容器,熟悉一下docker部署MySQL
- cd root/mysql/
- 方式1
-
docker run -id -p 3306:3306 \ --name=c_mysql2 \ -v $PWD/conf:/etc/mysql/conf.d \ -v $PWD/logs:/logs \ -v $PWD/data:/var/lib/mysql \ -e MYSQL_ROOT_PASSWORD=root \ mysql:5.7
- 方式2
-
docker run --name=c_mysql -d -p 3306:3306 -e MYSQL_ROOT_PASSWORD=root mysql:5.7
- 查看容器运行状态
-
docker ps -a
- 启动c_mysql容器
-
docker start c_mysql
- 到这一步的时候,出错了,新的MySQL容器和旧的MySQL容器都启动不了了,而且后来将新的容器删除之后,旧的容器依旧启动不了,也不知道啥情况(前面是用第一种方式创建,不行就请教了老师,改成了第二种方式)
- 然后到用Navicat连接诶数据库
-
然后现在开始创建项目,配置环境
- 自动创建springboot项目,导入依赖
-
<!--整合mybatis-plus--> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>5.1.49</version> </dependency> <dependency> <groupId>com.baomidou</groupId> <artifactId>mybatis-plus-boot-starter</artifactId> <version>3.0.5</version> </dependency> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.12</version> </dependency> <!--加密组件,apache--> <dependency> <groupId>commons-codec</groupId> <artifactId>commons-codec</artifactId> <version>1.3</version> </dependency> <!--redis依赖--> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId> </dependency> <!--jwt--> <dependency> <groupId>io.jsonwebtoken</groupId> <artifactId>jjwt</artifactId> <version>0.9.1</version> </dependency> <!--AMQP依赖--> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-amqp</artifactId> </dependency>
- 然后,根据我们的需要,我们现在需要在这个父项目当中,创建两个子模块(注意要创建maven项目,因为springboot项目本身就已经继承了一个父类,那么就会导致我们无法再继承刚刚我们创建的父项目)
- 在这两个模块当中,一个用来实现发送短消息,登录注册这三个功能,另一个用来实现收验证码的功能
- 新建模块1:Pubreg,手写启动类,加入核心配置文件,里面增加mq的配置
-
package com.pro; import org.mybatis.spring.annotation.MapperScan; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; @SpringBootApplication @MapperScan(value = "com.pro.mapper") public class PubRegApp { public static void main(String[] args) { SpringApplication.run(PubRegApp.class,args); } }
-
#mq spring.rabbitmq.host=192.168.8.171 #通信端口 spring.rabbitmq.port=5672 spring.rabbitmq.username=rabbit spring.rabbitmq.password=rabbit spring.rabbitmq.virtual-host=/
-
package com.pro.config; import org.springframework.amqp.core.Queue; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; //标识了下面这个注解,就代表是配置类:相当于以前的applicationContext.xml文件 @Configuration public class PubConfig { @Bean public Queue queue(){ return new Queue("yzmQueue"); } }
- 然后在这个模块里面开始写我们的发送验证码和登录业务
- 思路:业务层,我们需要根据前端传过来的用户名随机生成验证码,将用户名和密码存进reids,并且发送到yzmQueue队列中
-
package com.pro.service; import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; import com.baomidou.mybatisplus.extension.api.R; import com.pro.domain.User; import com.pro.mapper.UserMapper; import org.springframework.amqp.rabbit.core.RabbitTemplate; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.data.redis.core.RedisTemplate; import org.springframework.stereotype.Service; import java.time.Duration; import java.util.HashMap; import java.util.Map; import java.util.Random; @Service public class UserServiceImpl implements UserService { @Autowired private UserMapper userMapper; @Autowired private RedisTemplate redisTemplate; @Autowired private RabbitTemplate rabbitTemplate; /** * 发送验证码 * @param username */ @Override public void proYzm(String username) { Random random = new Random(); int yzm = random.nextInt(9000)+1000;//生成一个1000到10000之间的四位数验证码 //用用户名做键,验证码做值,存进Redis,并设置一分钟的存活时间 String YZM = String.valueOf(yzm); redisTemplate.opsForValue().set(username,YZM, Duration.ofSeconds(120)); //将yzmQueue作为队列名,验证码和手机号的作为消息发送到消息队列当中,前提是我们已经写了配置类,创建了这个队列 Map<Object,Object> map = new HashMap(); map.put("username",username); map.put("yzm",yzm); rabbitTemplate.convertAndSend("yzmQueue",map); } /** * 验证是否已经发送过验证码 * @param username * @return */ public String checkYzm(String username){ String redisData = (String) redisTemplate.opsForValue().get(username); return redisData; } @Override public String checkLogin(String username,String yzm) { String redisYzm = (String) redisTemplate.opsForValue().get(username); QueryWrapper queryWrapper = new QueryWrapper(); queryWrapper.eq("username",username); User user = userMapper.selectOne(queryWrapper); //如果用户不存在,注册并判断是否登录成功 if(user == null){ if(yzm.equals(redisYzm)){ User user1 = new User(); user1.setUsername(username); userMapper.insert(user1); return "尊贵的"+username+"用户,您已注册并登录成功!"; }else { return "验证码错误或已过期!"; } }else {//用户存在,直接判断是否可以登录 if(redisYzm.equals(yzm)){ return "尊贵的"+username+"用户,欢迎回来!"; }else { return "验证码错误或已过期!"; } } } }
package com.pro.controller; import com.pro.service.UserService; import com.pro.util.R; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.PostMapping; import org.springframework.web.bind.annotation.RestController; @RestController public class UserController { @Autowired private UserService userService; /** * 发验证码 * @param username * @return */ @PostMapping("/proYzm") public R proYzm (String username){ String checkYzm = userService.checkYzm(username); if(checkYzm == null){ userService.proYzm(username); return new R(200,"验证码发送成功!"); } return new R(300,"请两分钟之后再发验证码!"); } /** * 登录判断 * @param username * @param yzm * @return */ @PostMapping("/checkLogin") public String checkLogin(String username,String yzm){ String s = userService.checkLogin(username, yzm); return s; } }
- 效果:
- Redis里面可以看到信息只能存活我们设置的60秒
-
测试好发送验证码,我们现在来写接收验证码
- 首先建一个模块conReg,手写启动类,加入核心配置文件,里面增加mq的配置
- 写一个监听类来监听队列当中的验证码消息,模拟接收验证码
-
package com.pro.listener; import org.springframework.amqp.rabbit.annotation.RabbitListener; import org.springframework.stereotype.Component; import java.util.Map; @Component public class SpringRabbitListener { /* @RabbitListener(queues = "java") public void listenerQueue(String msg){ System.out.println("消费者接受到了消息:"+msg); } */ @RabbitListener(queues = "yzmQueue") public void listenerWorkQueue1(Map map)throws Exception{ System.out.println("【验证密码】您的验证码为:"+map.get("yzm")+"。尊敬的"+map.get("username")+"客户,以上验证密码2分钟内有效,请勿泄露或转发他人。【湖北移动 移动认证】"); } }
-
然后如果是已注册用户,登录会显示欢迎回来
-
验证码错误或者过期,就会提示验证码过期或错误,这里我没有设置具体是用户名错误还是验证码错误了。
- 遇到的问题
- 数据库ID没有设自动递增,插入user的时候,用null去调用get方法
- 后续需要改进:加入token,未登录不能操作,前端页面实现。