当前位置: 首页 > news >正文

讯联云库项目开发日志(一)

1、设计数据库

2、写基本框架

entity、controller、service、exception、utils、mapper

mapper层:

生成了一系列的CRUD方法

工具类:线程安全的日期工具类、 ​​参数校验工具类​

线程安全的日期工具类​​:主要用于 ​​日期格式化(format)和解析(parse)​​,并解决了 SimpleDateFormat 的线程安全问题​

参数校验工具类:主要用于检查对象参数是否满足“至少有一个非空字段”的条件,并提供了一些字符串辅助方法(如首字母大写、判空)

3、登录验证码校验

这段代码是一个 ​​验证码生成与校验​​ 的接口,主要用于生成图片验证码并存储到 HttpSession 中,以便后续验证用户输入的验证码是否正确 

  1. ​生成图片验证码​

    • 使用 CreateImageCode 类创建一个 ​​130x38 像素​​ 的验证码图片,包含 ​​5 个随机字符​​,干扰线数量为 ​​10​​。
    • 生成的验证码字符串存储在 code 变量中。
  2. ​设置 HTTP 响应头​

    • 禁用缓存,确保每次请求都生成新的验证码:
      response.setHeader("Pragma", "no-cache"); response.setHeader("Cache-Control", "no-cache"); response.setDateHeader("Expires", 0);
    • 设置响应类型为 image/jpeg,表示返回的是 JPEG 图片:
      response.setContentType("image/jpeg");
  3. ​存储验证码到 Session​

    • 根据 type 参数决定验证码的用途:
      • type=1:普通验证码(如登录验证),存储到 Constants.CHECK_CODE_KEY
      • 其他情况(如邮箱验证码),存储到 Constants.CHECK_CODE_KEY_EMAIL
  4. ​输出验证码图片​

    • 调用 vCode.write(response.getOutputStream()) 将生成的图片写入 HTTP 响应流,返回给前端显示。

为了辅助上述方法,因此创建了一个图形验证码生成工具类​​(CreateImageCode

package com.cjl.entity.dto;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;//验证码图片bufferprivate BufferedImage  buffImg=null;Random random=new Random();public CreateImageCode(){createCode();}public CreateImageCode(int width, int height, int codeCount, int lineCount){this.width=width;this.height=height;this.codeCount=codeCount;this.lineCount=lineCount;createCode();}public CreateImageCode(int width, int height, int lineCount){this.width=width;this.height=height;this.lineCount=lineCount;createCode();}public CreateImageCode(int width, int height){this.width=width;this.height=height;createCode();}//生成图片private void createCode(){int fontWidth=width/codeCount; //字体宽度int fontHeight=height-5; //字体高度int codeY=height-8; //验证码y坐标//创建图像bufferbuffImg=new BufferedImage(width,height,BufferedImage.TYPE_INT_RGB);Graphics g=buffImg.getGraphics();//设置背景颜色g.setColor(getRandColor(200,250));g.fillRect(0,0,width,height);//设置字体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是坐标,基于要画的东西最左侧字符的基线g.drawString(strRand,i*fontWidth+3,codeY);}}//得到随机字符private String RandomStr(int n){String str1="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";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 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/period+ (6.2831853071795862D* (double)phase/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();}
}

最终达到的效果:

4、创建邮箱数据库

依旧通过java生成器生成,不过,第一次遇见这种:

	<!-- 通用查询结果列--><sql id="base_column_list">e.email,e.code,e.creat_time,e.status</sql><sql id="base_condition_filed"><if test="query.email != null and query.email!=''">and  e.email = #{query.email}</if><if test="query.code != null and query.code!=''">and  e.code = #{query.code}</if><if test="query.creatTime != null and query.creatTime!=''"><![CDATA[ and  e.creat_time=str_to_date(#{query.creatTime}, '%Y-%m-%d') ]]></if><if test="query.status != null">and  e.status = #{query.status}</if></sql><!-- 通用条件列--><sql id="base_condition"><where><include refid="base_condition_filed" /></where></sql><!-- 通用查询条件列--><sql id="query_condition"><where><include refid="base_condition_filed" /><if test="query.emailFuzzy!= null  and query.emailFuzzy!=''">and  e.email like concat('%', #{query.emailFuzzy}, '%')</if><if test="query.codeFuzzy!= null  and query.codeFuzzy!=''">and  e.code like concat('%', #{query.codeFuzzy}, '%')</if><if test="query.creatTimeStart!= null and query.creatTimeStart!=''"><![CDATA[ and  e.creat_time>=str_to_date(#{query.creatTimeStart}, '%Y-%m-%d') ]]></if><if test="query.creatTimeEnd!= null and query.creatTimeEnd!=''"><![CDATA[ and  e.creat_time< date_sub(str_to_date(#{query.creatTimeEnd},'%Y-%m-%d'),interval -1 day) ]]></if></where></sql>

 <sql id="base_column_list"> ???我很好奇了这是什么

通过 <sql> 标签定义可重用的 SQL 片段,并通过 <include> 标签引用这些片段,从而提高代码的复用性和可维护性。

当 MyBatis 启动时,会解析这些 <sql> 片段并存储在内存中,形成可重用的 SQL 模板。

1. ​​片段注册​
  • base_column_list → 字段列表模板
  • base_condition_filed → 基础条件模板
  • base_condition → 完整 WHERE 条件(直接引用 base_condition_filed
  • query_condition → 扩展 WHERE 条件(引用 base_condition_filed 并追加模糊/范围查询)
2.逻辑关系

这段代码的核心思想就是 ​​将常用的 SQL 片段拆解成可复用的模块​​,通过 MyBatis 的 <sql> 和 <include> 机制实现 ​​逻辑复用​​ 和 ​​动态拼接​​。但它不仅仅是简单的“代码片段集合”,而是一种 ​​模块化 SQL 设计模式​

可能跟我一样第一次遇见的同学看见这里就有点蒙圈了,没事,我们来举例 一个简单的代码来理解:

1. 定义可复用的 SQL 片段(乐高积木块)
<!-- 基础车体(相当于字段列表) -->
<sql id="base_car_body">id, brand, model, color
</sql><!-- 基础轮子(相当于基础条件) -->
<sql id="base_wheels"><if test="wheelSize != null">AND wheel_size = #{wheelSize}</if>
</sql>
2.组装小车(简单查询)
<select id="selectBasicCar" resultType="Car">SELECT <include refid="base_car_body"/>  <!-- 插入车体 -->FROM cars<where><include refid="base_wheels"/>    <!-- 插入轮子条件 --></where>
</select>
3.生成的SQL​​(当 wheelSize=18 时):
SELECT id, brand, model, color
FROM cars
WHERE wheel_size = 18

这样是不是就好理解一点了!!

5、实现发送邮箱验证码接口 

就在主播写这个接口的发送邮箱验证的时候,命名mappers互相跳转都没有问题,困扰我一个多小时,结果发现没有在​​属性配置文件添加mybatis.mapper-locations=classpath:mappers/*.xml,因此虽然我可以互相跳转,但是代码自己找不到sql映射,属所以说写代码还是要规范

言归正传,

当我们写完这个接口,我们要考虑很多因素,首先就是用户输入的验证码是不是正确的,要根据之前定义的checkCode方法来查看,其次就要考虑status的问题,如果账户已经被注册了就没有必要了要进行判断,不仅如此,当用户多次点击发送的时候,我们只需要禁用之前的验证码,这个如何实现?

答案是在service层

@Update("update email_code set status=1 where email=#{email} and status=0")

为什么这样就可以?我们来分析,当检测到用户未注册的时候,代码的流程走到这里来的时候,此时他的status就会变成1,也就是说每次发送新验证码前,会先执行 disableEmailCode 方法,将所有该邮箱未使用的验证码(status=0)标记为被使用了已禁用(status=1)。因此能保证最晚(最新)的那一个验证码才会生效

我们现在来发送邮件, ​​JavaMailSender​​(Spring框架提供的邮件发送工具)来创建并发送一封简单的电子邮件:

流程:

1.入口​​:调用sendMailCode(email, code)方法,传入收件人邮箱和验证码

2.获取系统邮件模板

SysSettingDto sysSettingDto = redisComponent.getSysSetting();

设置邮件发送的格式: 

3.构建邮件内容
  • ​标题处理​​:直接使用系统配置的标题

    helper.setSubject(sysSettingDto.getRegisterMailTitle());

  • ​内容格式化​​:将验证码插入模板
    String.format("您好,您的邮箱验证码为:%s,15分钟有效", "A1B2C3")

提问:有人就好奇了,redis一开始是空的哪来的模板??

答案是 

1. 尝试从Redis读取(此时返回null),发现为空时,创建默认配置

最后保存到Redis(无过期时间) 

整个发邮件的大致过程就是这样,最后用户会收到:

相关文章:

  • 3.2 一点一世界
  • 嵌入式学习笔记 - HAL_ADC_ConfigChannel函数解析
  • 出于PCB设计层面考虑,连排半孔需要注意哪些事项?
  • 构建媲美 ChatGPT 的 AI 交互界面—OpenWebUI
  • Flannel UDP 模式的优缺点
  • WebRTC技术EasyRTC嵌入式音视频通信SDK打造远程实时视频通话监控巡检解决方案
  • JPG与PDF格式转换器
  • 06 mysql之DML
  • R-tree详解
  • 2025年第十六届蓝桥杯大赛软件赛C/C++大学B组题解
  • C++设计模式——单例模式
  • SpringBoot 3.X 开发自己的 Spring Boot Starter 和 SpringBoot 2.x 的区别
  • Python查询ES错误ApiError(406, ‘Content-Type ...is not supported
  • 【传感器多模态融合与AI驱动】
  • Leetcode 3548. Equal Sum Grid Partition II
  • 常见网卡接口命名方式
  • OpenAI for Countries:全球AI基础设施的“技术基建革命”
  • 苍穹外卖--新增菜品
  • 新华三H3CNE网络工程师认证—路由参数与比较
  • Gartner《如何有效融合Data Fabric 与Data Mesh数据战略》学习心得
  • 大英博物馆展歌川广重:他是梵高最钟爱的浮世绘名家
  • 《蛮好的人生》:为啥人人都爱这个不完美的“大女主”
  • 西北大学副校长范代娣成陕西首富?系家庭财富,本人已从上市公司退出
  • 人民日报访巴西总统卢拉:“巴中关系正处于历史最好时期”
  • 金价大跌!足金饰品每克一夜便宜14元,涨势是否已终结?
  • 季子文化与江南文化的根脉探寻与融合