注册流程
【1】前端提交注册信息
【2】后端接受数据
【3】后端生成一个UUID作为token,将token做为redis的key值,用户数据做为redis的value值,并设置key的时长
【4】后端根据用户信息中的邮箱地址信息,检验用户是否已经注册,若是没有,生成注册连接发送到用户邮箱,若是已经注册,提示用户该邮箱地址已被注册
【5】用户点击邮件中的注册连接
【6】后端判断redis中token是否过时,没有将用户信息保存到数据库,提示用户注册成功
项目源码:https://gitee.com/residual-temperature/email-link-demo.git
邮箱效果图
html
一、pom文件要加入的jar包前端
<!-- 邮件相关 --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-mail</artifactId> </dependency> <!-- redis相关 --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId> </dependency>
二、application.yml文件中要加入的配置java
spring: redis: host: # redis地址 port: 6379 # redis端口号(默认6379) password: # redis密码 mail: host: smtp.qq.com # 邮箱协议 username: 地址 # 发送的邮箱地址 password: 受权码 # 邮箱的受权码
三、定义实体类git
@Repository @Data @AllArgsConstructor @NoArgsConstructor public class User implements Serializable { private long id; private String account; private String password; private String username; }
注意redis
此处没有get(),set()方法是由于导入了lombok包spring
四、redis的config配置
对象的保存须要序列化,因此须要自定义RedisTempletesql
@Configuration public class RedisConfig { //编写本身的配置类 @Bean @SuppressWarnings("all") public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory) { //为了开发方便通常使用<String,Object> RedisTemplate<String, Object> template = new RedisTemplate<>(); template.setConnectionFactory(redisConnectionFactory); //JSON序列化的配置 Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class); ObjectMapper om = new ObjectMapper(); om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY); om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL); jackson2JsonRedisSerializer.setObjectMapper(om); //String的序列化 StringRedisSerializer stringRedisSerializer = new StringRedisSerializer(); //key采用String的序列化方式 template.setKeySerializer(stringRedisSerializer); //hash采用String的序列方式 template.setHashKeySerializer(stringRedisSerializer); //value序列化采用jackson template.setValueSerializer(jackson2JsonRedisSerializer); //hash的Value序列化采用jackson template.setHashValueSerializer(jackson2JsonRedisSerializer); template.afterPropertiesSet(); return template; } }
五、验证连接生成和邮箱发送的工具类CodeUtils的配置数据库
@Component public class CodeUtils { @Resource JavaMailSender mailSender; @Resource RedisTemplate<String, User> redisTemplate; // 生成连接,并给接收的邮箱发送邮件 public boolean sendCode(User user){ MimeMessage message = mailSender.createMimeMessage(); try{ MimeMessageHelper messageHelper = new MimeMessageHelper(message); String token = UUID.randomUUID().toString(); // 生成UUID redisTemplate.opsForValue().set(token,user); redisTemplate.expire(token,300, TimeUnit.SECONDS); messageHelper.setFrom("发送方的邮箱地址"); //发送方的邮箱地址,而不是接收方的邮箱地址 messageHelper.setTo(user.getAddress()); // 接收方的邮箱地址 messageHelper.setSubject("注册"); // 邮箱标题 String html = "<html>\n" + "<body>\n" + "<p>请点击下方连接注册</p>\n" + "<a href=\"http://localhost:8081/lookCode/"+token+"\">http://localhost:8081/lookCode/"+token+"</a>" + "</body>\n" + "</html>"; messageHelper.setText(html,true); // 邮箱内容 mailSender.send(message); // 发送邮箱 System.out.println("发送成功"); return true; }catch (Exception e){ System.out.println("发送失败"); return false; } } // 判断token是否过时 public boolean eqToken(String token){ return redisTemplate.hasKey(token); } // 根据token查询用户的信息 public User findUser(String token){ return redisTemplate.opsForValue().get(token); } }
六、UserMapper的配置后端
@Mapper @Repository public interface UserMapper { // 添加用户 注解开发sql语句 @Insert("insert into user(account,password,username) values (#{account},#{password},#{username})") public int addUser(User user); }
七、UserService的配置网络
public interface UserService { // 添加用户 public boolean adduser(User user); // 根据用户注册信息进行注册连接的的生成和发送 public boolean sendCode(User user); // 用户点击注册连接判断token是否过时 public boolean eqToken(String token); }
八、UserService的实现类UserServiceImpl的配置
@Service public class UserServiceImpl implements UserService { @Resource UserMapper userMapper; @Resource CodeUtils codeUtils; /** * 添加注册的用户信息 * @param user 注册的用户信息 * @return 是否添加成功 */ @Override public boolean adduser(User user) { return userMapper.addUser(user) > 0; } /** * 生成连接和发送连接 * @param address 接收的邮箱地址 * @param user 注册的用户信息 */ @Override public boolean sendCode(User user) { if ( codeUtils.sendCode(user)) // 调用验证连接生成工具类中的生成连接和发送邮件函数 return true; else return false; } /** * 判断token是否过时 * @param token 用户注册所接收的token * @return 注册成功与否 */ @Override public boolean eqToken(String token) { boolean flag = codeUtils.eqToken(token); if (flag){ User user = codeUtils.findUser(token); adduser(user); return true; }else { return false; } } }
九、UserController的配置
@RestController public class UserController { @Resource UserService userService; // 根据用户注册信息进行注册连接的的生成和发送 @PostMapping("/sendCode") public Map<String,String> sendCode(@RequestBody User user){ boolean flag = userService.sendCode(user); Map<String,String> map = new HashMap<>(); if (flag){ map.put("msg","邮件发送成功,请前往您的邮箱进行注册验证"); return map; }else { map.put("msg","邮件发送失败"); return map; } } // 判断是否注册成功 @GetMapping("/lookCode/{token}") public Map<String,String> lookCode(@PathVariable("token")String token){ boolean flag = userService.eqToken(token); Map<String,String> map = new HashMap<>(); if (flag){ map.put("msg","注册成功"); /* 后续的操做 ... ...*/ return map; }else { map.put("msg","注册码过时,请从新注册"); return map; } } }
由于没有写前端页面,因此就用postman和页面来演示
传入user对象
返回结果
邮箱连接
点击注册连接以后
注册成功以后数据库先后对比
注册成功以前
注册成功以后
可能会遇到的问题 【1】有些内部网络不支持发送邮箱,若是保证代码没错,能够换个网络试试 【2】若是是在本地测试,链接的是本地redis,记得开启本地的redis