网盘登录业务记录

一、获取验证码

本项目中利用文件流写回验证码图片。

工具类核心:利用jdk内部的BufferedImage类:这个类生成一个图片存到内存缓冲区,方便我们对图片进行操作。

package com.easypan.utils;

import javax.imageio.ImageIO;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.IOException;
import java.io.OutputStream;
import java.util.Random;

public class CreateImageCode {
    // 图片的宽度。
    private int width = 160;
    // 图片的高度。
    private int height = 40;
    // 验证码字符个数
    private int codeCount = 4;
    // 验证码干扰线数
    private int lineCount = 20;
    // 验证码
    private String code = null;
    // 验证码图片Buffer
    private BufferedImage buffImg = null;
    Random random = new Random();

    public CreateImageCode() {
        creatImage();
    }

    public CreateImageCode(int width, int height) {
        this.width = width;
        this.height = height;
        creatImage();
    }

    public CreateImageCode(int width, int height, int codeCount) {
        this.width = width;
        this.height = height;
        this.codeCount = codeCount;
        creatImage();
    }

    public CreateImageCode(int width, int height, int codeCount, int lineCount) {
        this.width = width;
        this.height = height;
        this.codeCount = codeCount;
        this.lineCount = lineCount;
        creatImage();
    }

    // 生成图片
    private void creatImage() {
        int fontWidth = width / codeCount;// 字体的宽度
        int fontHeight = height - 5;// 字体的高度
        int codeY = height - 8;

        // 图像buffer
        buffImg = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
        Graphics g = buffImg.getGraphics();
        //Graphics2D g = buffImg.createGraphics();
        // 设置背景色
        g.setColor(getRandColor(200, 250));
        g.fillRect(0, 0, width, height);
        // 设置字体
        //Font font1 = getFont(fontHeight);
        Font font = new Font("Fixedsys", Font.BOLD, fontHeight);
        g.setFont(font);

        // 设置干扰线
        for (int i = 0; i < lineCount; i++) {
            int xs = random.nextInt(width);
            int ys = random.nextInt(height);
            int xe = xs + random.nextInt(width);
            int ye = ys + random.nextInt(height);
            g.setColor(getRandColor(1, 255));
            g.drawLine(xs, ys, xe, ye);
        }

        // 添加噪点
        float yawpRate = 0.01f;// 噪声率
        int area = (int) (yawpRate * width * height);
        for (int i = 0; i < area; i++) {
            int x = random.nextInt(width);
            int y = random.nextInt(height);
            buffImg.setRGB(x, y, random.nextInt(255));
        }

        String str1 = randomStr(codeCount);// 得到随机字符
        this.code = str1;
        for (int i = 0; i < codeCount; i++) {
            String strRand = str1.substring(i, i + 1);
            g.setColor(getRandColor(1, 255));
            // g.drawString(a,x,y);
            // a为要画出来的东西,x和y表示要画的东西最左侧字符的基线位于此图形上下文坐标系的 (x, y) 位置处

            g.drawString(strRand, i * fontWidth + 3, codeY);
        }
    }

    // 得到随机字符
    private String randomStr(int n) {
        String str1 = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz1234567890";
        String str2 = "";
        int len = str1.length() - 1;
        double r;
        for (int i = 0; i < n; i++) {
            r = (Math.random()) * len;
            str2 = str2 + str1.charAt((int) r);
        }
        return str2;
    }

    // 得到随机颜色
    private Color getRandColor(int fc, int bc) {// 给定范围获得随机颜色
        if (fc > 255) fc = 255;
        if (bc > 255) bc = 255;
        int r = fc + random.nextInt(bc - fc);
        int g = fc + random.nextInt(bc - fc);
        int b = fc + random.nextInt(bc - fc);
        return new Color(r, g, b);
    }

    /**
     * 产生随机字体
     */
    private Font getFont(int size) {
        Random random = new Random();
        Font font[] = new Font[5];
        font[0] = new Font("Ravie", Font.PLAIN, size);
        font[1] = new Font("Antique Olive Compact", Font.PLAIN, size);
        font[2] = new Font("Fixedsys", Font.PLAIN, size);
        font[3] = new Font("Wide Latin", Font.PLAIN, size);
        font[4] = new Font("Gill Sans Ultra Bold", Font.PLAIN, size);
        return font[random.nextInt(5)];
    }

    // 扭曲方法
    private void shear(Graphics g, int w1, int h1, Color color) {
        shearX(g, w1, h1, color);
        shearY(g, w1, h1, color);
    }

    private void shearX(Graphics g, int w1, int h1, Color color) {

        int period = random.nextInt(2);

        boolean borderGap = true;
        int frames = 1;
        int phase = random.nextInt(2);

        for (int i = 0; i < h1; i++) {
            double d = (double) (period >> 1) * Math.sin((double) i / (double) period + (6.2831853071795862D * (double) phase) / (double) frames);
            g.copyArea(0, i, w1, 1, (int) d, 0);
            if (borderGap) {
                g.setColor(color);
                g.drawLine((int) d, i, 0, i);
                g.drawLine((int) d + w1, i, w1, i);
            }
        }

    }

    private void shearY(Graphics g, int w1, int h1, Color color) {

        int period = random.nextInt(40) + 10; // 50;

        boolean borderGap = true;
        int frames = 20;
        int phase = 7;
        for (int i = 0; i < w1; i++) {
            double d = (double) (period >> 1) * Math.sin((double) i / (double) period + (6.2831853071795862D * (double) phase) / (double) frames);
            g.copyArea(i, 0, 1, h1, 0, (int) d);
            if (borderGap) {
                g.setColor(color);
                g.drawLine(i, (int) d, i, 0);
                g.drawLine(i, (int) d + h1, i, h1);
            }

        }

    }

    public void write(OutputStream sos) throws IOException {
        ImageIO.write(buffImg, "png", sos);
        sos.close();
    }

    public BufferedImage getBuffImg() {
        return buffImg;
    }

    public String getCode() {
        return code.toLowerCase();
    }
}

controller层: 

 @RequestMapping("/checkCode")
    public void checkCode(HttpServletResponse response, HttpSession session,Integer type) throws IOException {
        //验证码的高度、宽度、字符的数量、干扰线的数量
        CreateImageCode vCode = new CreateImageCode(130, 38, 4, 10);
        response.setHeader("Pragma", "no-cache");
        response.setHeader("Cache-Control", "no-cache");
        response.setDateHeader("Expires", 0);
        //上面三行。禁止缓存是为了确保验证码图片每次都是新生成的,而不是从缓存中获取旧的验证码图片。
        response.setContentType("image/jpeg");
        //将验证码存到session
        String code = vCode.getCode();
        //将生成的数据存到session
        if (type == null || type == 0) {
            //0是登录
            session.setAttribute(Constants.CHECK_CODE_KEY, code);
        } else {
            //注册
            session.setAttribute(Constants.CHECK_CODE_KEY_EMAIL, code);
        }

 解释:每次访问页面显示的验证码都要刷新,下面的代码是在服务端禁用jsp缓存。 Cache-Control头域用来指定请求和响应遵循的缓存机制。

response.setHeader("Pragma", "no-cache");
response.setHeader("Cache-Control", "no-cache");
response.setDateHeader("Expires", 0);

设置返回值类型:字节流

response.setContentType("image/jpeg");

  前端接收 字节流对象。

业务中有好几处需要用到验证码:

1.为什么要将这验证码分成两类存储,(session中对应不同的key)???

注册业务需要同时用到两处的验证码。普通验证码和邮箱验证码,进行比对时都要比对。

补充:也可以后端返回前端base64编码,前端接收展示图片。验证码校验可以存到session中也可以存到redis中。 

二、邮箱发送验证码(注册和忘记密码找回时使用)

利用MimeMessage    javaMailSender  MimeMessageHelper发送邮件

参数type:0表示注册,1表示找回密码

逻辑:1.如果type是0,在发送邮件之前判断该邮箱是否被占用。2.发送生成的随机数

3.将发送的验证码中其它的设置成失效。4.将刚才发送的验证码和email信息存到数据库。

这个数据库表存在的意义??

如果有两个程序并发执行,输入同一个email,如果不存到数据库中,两个都会通过,引发异常。

相关依赖 :

  <!--邮件发送-->
	    <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-mail</artifactId>
            <version>${springboot.version}</version>
        </dependency>

配置文件: 

# 配置邮件服务器的地址 smtp.qq.com
spring.mail.host=smtp.qq.com
# 配置邮件服务器的端口(465或587)
spring.mail.port=465
# 配置用户的账号
spring.mail.username=xxxxxxxxxxx@qq.com
# 配置用户的密码
spring.mail.password=xxxxxxxxxxx
# 配置默认编码
spring.mail.default-encoding=UTF-8
# SSL 连接配置
spring.mail.properties.mail.smtp.socketFactory.class=javax.net.ssl.SSLSocketFactory
# 开启 debug,这样方便开发者查看邮件发送日志
spring.mail.properties.mail.debug=true

 Spring Boot的starter模块提供了自动化配置,所以在引入了spring-boot-starter-mail依赖之后,会根据配置文件中的内容去创建JavaMailSender实例 

可以查看一下JavaMailSender的视线类:

    private void sendCode(String ToEmail, String s) {
        try {
            MimeMessage message= javaMailSender.createMimeMessage();
            MimeMessageHelper helper=new MimeMessageHelper(message,true);
            helper.setFrom(appConfig.getSendUserName());
            helper.setTo(ToEmail);
            SysSettingsDto sysSettingsDto = redisComponent.getSysSettingsDto();
            helper.setSubject(sysSettingsDto.getRegisterEmailTitle());//标题
            helper.setText(String.format(sysSettingsDto.getRegisterEmailContent(),s));
            //内容
            helper.setSentDate(new Date());//日期
            javaMailSender.send(message);//发送
        }catch (Exception e){
    
        }
    }

事务 

@Transactional(rollbackFor = Exception.class)注解,spring提供的声明式事务管理注解。

发送邮箱验证码需要做事务管理的原因是确保邮件发送的可靠性和一致性。在发送邮件的过程中,涉及到多个步骤,如创建邮件、设置邮件内容、发送邮件等。如果其中某个步骤出现异常或失败,事务管理可以确保在出现问题时回滚所有操作,避免发送不完整或重复的邮件。事务管理可以保证邮件发送的可靠性,确保邮件要么完全发送成功,要么完全失败,从而维护系统的数据一致性。

aop做参数校验

业务中不做参数校验可以出现空指针的严重错误

1.定义校验注解

2.创建切面类,编写校验逻辑。

发送邮件

MimeMessage message= javaMailSender.createMimeMessage();
MimeMessageHelper helper=new MimeMessageHelper(message,true);
helper.setFrom(appConfig.getSendUserName());
helper.setTo(ToEmail);
SysSettingsDto sysSettingsDto = redisComponent.getSysSettingsDto();
helper.setSubject(sysSettingsDto.getRegisterEmailTitle());
helper.setText(String.format(sysSettingsDto.getRegisterEmailContent(),s));
helper.setSentDate(new Date());
javaMailSender.send(message);

MimeMessage表示一个mail对象。

MimeMessageHelper类辅助设置发送邮件的属性。

为什么将发送邮件的模版存到redis中?

由于发送邮件涉及到第三方服务、响应时间可能会过长。这是个耗时操作、如果放在用户请求进程中同步处理,响应时间过长。为了提升用户体验,存入redis中。(也可以使用消息队列)

  • 14
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值