【苍穹外卖】学习日志-day1

embedded/2024/11/20 0:05:09/

目录

nginx 反向代理介绍

nginx 的优势

提高访问速度

负载均衡

保证后端服务安全

高并发静态资源

Swagger 生成 API 文档

Swagger 的使用方式

导入knife4j的maven坐标

在配置类中加入knife4j相关配置

 设置静态资源映射

通过注解控制生成的接口文档

项目技术点

Token 模式

 MD5 加密

新增员工开发

DTO 设计模式

隔离线程 ThreadLocal

全局异常处理器

为什么需要全局异常处理器

代码测试

员工分页查询

代码测试


本章代码地址:苍穹外卖

 

nginx 反向代理介绍

        在没学习 nginx 之前,我们的项目都是前端直接发请求 tomcat 服务器的。如下图

tomcat 的作用:Tomcat 可以处理 HTTP 请求并将其传递给 Java 应用程序进行处理。

        学习了 nginx 后,我们更希望将用户的所有请求交给 nginx 反向代理,再转发给 tomcat 处理:

nginx 的优势

提高访问速度

        nginx 可以做缓存,如果我们请求的是同一个接口地址,则无需请求后端服务,直接在 nginx 把缓存数据响应给前端。

负载均衡

        nginx 可以把大量的请求按照指定的方式均衡的分配给集群中的每台服务器


举个例子:

        以百度为例,百度的后台肯定是不止一台服务器的,但我们在访问百度的时候,只需要输入百度的地址,就会被分配到一个服务器上去,以获得服务。而我们访问的是哪个服务器我们并不知道,我们只管访问 www.baidu.com,后面的事都会有相应的机制帮我们实现。

        要实现此类效果,即无论应用有多少实例,我们只需要访问一个地址就可以得到服务。就需要在客户端与服务端之间加上一层服务器 nginx 。

        这样客户端只管访问 nginx ,再由 nginx 服务器将请求代理到真正部署有实例的服务器上去即可。

保证后端服务安全

        我们真实的服务器不应该直接暴露到公网上去,否则更加容易泄露服务器的信息,也更加容易受到攻击。而使用 nginx 可以接收来自客户端的请求并将其转发到后端服务器。这样做的好处是可以隐藏服务器的真实 IP 地址,提供额外的安全层

高并发静态资源

        nginx 专注于处理静态资源,具有出色的性能和高并发处理能力。将 nginx 作为静态资源服务器可以提高系统的响应速度,并减轻 tomcat 的负担。

         而苍穹外卖也是通过 nginx 反向代理来访问我们后台的,nginx 文件:nginx反向代理

Swagger 生成 API 文档

        使用 Swagger 你只需要按照它的规范去定义接口及接口相关的信息,就可以做到生成接口文档,以及在线接口调试页面。这样我们就不用去 Postman 配置路径再来测试了。

Knife4j 是为 Java MVC 框架集成 Swagger 生成 API 文档的增强解决方案~

Swagger 的使用方式

导入knife4j的maven坐标

            <dependency><groupId>com.github.xiaoymin</groupId><artifactId>knife4j-spring-boot-starter</artifactId><version>3.0.2</version></dependency>

在配置类中加入knife4j相关配置

WebMvcConfiguration

  • Swagger 实例Bean是Docket,所以通过配置Docket实例来配置Swaggger
  • Docket 实例关联上 apiInfo
    /*** 通过knife4j生成接口文档*/@Beanpublic Docket docket() {ApiInfo apiInfo = new ApiInfoBuilder().title("苍穹外卖项目接口文档")  // 生成标题.version("2.0")              // 版本.description("苍穹外卖项目接口文档").build();Docket docket = new Docket(DocumentationType.SWAGGER_2).apiInfo(apiInfo)// 通过select()方法,去配置扫描接.select()// RequestHandlerSelectors 配置如何扫描接口.apis(RequestHandlerSelectors.basePackage("com.sky.controller")).paths(PathSelectors.any()).build();return docket;}

 设置静态资源映射

    /*** 设置静态资源映射*/protected void addResourceHandlers(ResourceHandlerRegistry registry) {log.info("开始设置静态资源映射...");registry.addResourceHandler("/doc.html").addResourceLocations("classpath:/META-INF/resources/");registry.addResourceHandler("/webjars/**").addResourceLocations("classpath:/META-INF/resources/webjars/");}

通过注解控制生成的接口文档

比如

    @PostMapping@ApiOperation("新增员工")public Result save(@RequestBody EmployeeDTO employeeDTO) {// ...}
注解说明
@Api用在类上,例如Controller,表示对类的说明
@ApiModel用在类上,例如 entity、DTO、VO
@ApiModelProperty用在属性上,描述属性信息
@ApiOperation用在方法上,例如Controller的方法,说明方法的用途、作用 

当程序运行时,我们访问 http://localhost:8080/doc.html 便可访问我们生成的 Swagger 接口文档。

 

项目技术点


Token 模式

        JWT 是 JSON Web Token 的缩写,即 JSON Web 令牌,JWT 是通过对JSON进行加密签名来实现授权验证的方案,就是登陆成功后将相关信息组成 json 对象,然后对这个对象进行某中方式的加密,返回给客户端,客户端在下次请求时带上这个 token,服务端再收到请求时校验 token 合法性,其实也是在校验请求的合法性,只有通过校验成功才能访问后台。 

        所以 JWT 常用于完成客户端的登入系统,以及拦截器;只有当用户输入的账号与密码与数据库中的匹配,系统就会生成 JWT 令牌。而只有这个令牌,后台的接口的拦截器才会放行。

 JWT 依赖:

    <dependency><groupId>com.auth0</groupId><artifactId>java-jwt</artifactId><version>0.9.1</version></dependency>

  JWT 令牌生成工具类:

public class JwtUtil {/*** 生成jwt* 使用Hs256算法, 私匙使用固定秘钥** @param secretKey jwt秘钥* @param ttlMillis jwt过期时间(毫秒)* @param claims    设置的信息* @return*/public static String createJWT(String secretKey, long ttlMillis, Map<String, Object> claims) {// 指定签名的时候使用的签名算法,也就是header那部分SignatureAlgorithm signatureAlgorithm = SignatureAlgorithm.HS256;// 生成JWT的时间long expMillis = System.currentTimeMillis() + ttlMillis;Date exp = new Date(expMillis);// 设置jwt的bodyJwtBuilder builder = Jwts.builder()// 如果有私有声明,一定要先设置这个自己创建的私有的声明,这个是给builder的claim赋值,一旦写在标准的声明赋值之后,就是覆盖了那些标准的声明的.setClaims(claims)// 设置签名使用的签名算法和签名使用的秘钥.signWith(signatureAlgorithm, secretKey.getBytes(StandardCharsets.UTF_8))// 设置过期时间.setExpiration(exp);return builder.compact();}/*** Token解密** @param secretKey jwt秘钥 此秘钥一定要保留好在服务端, 不能暴露出去, 否则sign就可以被伪造, 如果对接多个客户端建议改造成多个* @param token     加密后的token* @return*/public static Claims parseJWT(String secretKey, String token) {// 得到DefaultJwtParserClaims claims = Jwts.parser()// 设置签名的秘钥.setSigningKey(secretKey.getBytes(StandardCharsets.UTF_8))// 设置需要解析的jwt.parseClaimsJws(token).getBody();return claims;}}

拦截器

/*** jwt令牌校验的拦截器*/
@Component
@Slf4j
public class JwtTokenAdminInterceptor implements HandlerInterceptor {@Autowiredprivate JwtProperties jwtProperties;/*** 校验jwt*/public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {//判断当前拦截到的是Controller的方法还是其他资源if (!(handler instanceof HandlerMethod)) {//当前拦截到的不是动态方法,直接放行return true;}//1、从请求头中获取令牌String token = request.getHeader(jwtProperties.getAdminTokenName());//2、校验令牌try {log.info("jwt校验:{}", token);Claims claims = JwtUtil.parseJWT(jwtProperties.getAdminSecretKey(), token);Long empId = Long.valueOf(claims.get(JwtClaimsConstant.EMP_ID).toString());BaseContext.setCurrentId(empId);log.info("当前员工id:", empId);//3、通过,放行return true;} catch (Exception ex) {//4、不通过,响应401状态码response.setStatus(401);return false;}}

可以通过 yml 配置文件来定义 JWT 令牌的一些属性

sky:jwt:# 设置jwt签名加密时使用的秘钥admin-secret-key: itcast# 设置jwt过期时间admin-ttl: 7200000# 设置前端传递过来的令牌名称admin-token-name: token

 

 MD5 加密

        当我们需要保存某些密码信息以用于身份确认时,如果直接将密码信息以明码方式保存在数据库中,不使用任何保密措施,系统管理员就很容易能得到原来的密码信息,这些信息一旦泄露, 密码也很容易被破译。

        为了增加安全性,有必要对数据库中需要保密的信息进行加密,MD5 算法可以很好地解决这个问题,因为它可以将任意长度的输入串经过计算得到固定长度的输出,而且只有在明文相同的情况下,才能等到相同的密文,并且这个算法是不可逆,即便得到了加密以后的密文,也不可能通过解密算法反算出明文。

 spring 提供了一个工具类 DigestUtils,我们可以利用该工具类来对数据进行加密

依赖坐标

<dependency><groupId>commons-codec</groupId><artifactId>commons-codec</artifactId><version>1.15</version></dependency>

 员工登入代码

    public Employee login(EmployeeLoginDTO employeeLoginDTO) {String username = employeeLoginDTO.getUsername();String password = employeeLoginDTO.getPassword();//1、根据用户名查询数据库中的数据Employee employee = employeeMapper.getByUsername(username);//2、处理各种异常情况(用户名不存在、密码不对、账号被锁定)if (employee == null) {//账号不存在throw new AccountNotFoundException(MessageConstant.ACCOUNT_NOT_FOUND);}//密码比对// TODO 后期需要进行md5加密,然后再进行比对// 对输入密码进行加密password = DigestUtils.md5DigestAsHex(password.getBytes());log.info(password);// 输入密码与数据库密码做比较if (!password.equals(employee.getPassword())) {//密码错误throw new PasswordErrorException(MessageConstant.PASSWORD_ERROR);}if (employee.getStatus() == StatusConstant.DISABLE) {//账号被锁定throw new AccountLockedException(MessageConstant.ACCOUNT_LOCKED);}//3、返回实体对象return employee;}

        以上是将用户输入的密码通过 DigestUtils 工具类转化成密文再与数据中的密码进行匹配,注意数据库中的密码存储的是密文。这样即使是数据库的管理者,也不能知道用户的真实密码。

password = DigestUtils.md5DigestAsHex(password.getBytes());

新增员工开发


DTO 设计模式

        数据传输对象 DTO 是一种设计模式,用于封装和传输应用程序不同层之间的数据。

        DTO 是轻量级对象,通常只包含必要的字段,不包含任何业务逻辑。DTO作用于应用程序中不同的业务之间的数据传输,例如在前端和后端之间或在分布式系统中不同的微服务之间。
        在 Spring Boot 应用程序中,DTO 特别有用,因为需要在控制器层、服务层和持久层之间传输数据。通过使用 DTO 就可以将内部数据模型与外部表示
解耦,从而更好地控制数据传输。

         如上图,前端传入这么一组数据,如果有那么一两个数据不是这个实体类 Employee 的,这个时候我们就不能直接用 EmployeeDTO 来接收;而是创建一个 DTO 类 EmployeeDTO 并在数据的 Service 层对 EmployeeDTO 与 Employee 中的共同属性进行赋值操作。这样前端传入的不同属性就不会对 Employee 产生影响,起到一定的解耦效果。

        DTO 允许将暴露给外部的数据与内部的模型隔离。这可以防止暴露敏感和不必要的数据,并为数据交换提供清晰的字段,也在一定程度上保证了内部数据的安全性。


Controller

    /** @description:新增员工**/@PostMapping@ApiOperation("新增员工")public Result save(@RequestBody EmployeeDTO employeeDTO) {log.info("新增员工:{}",employeeDTO);employeeService.save(employeeDTO);return Result.success();}

Service

        由于 EmployeeDTO 的属于与 Employee 的属性都是共同属性,所以可以直接使用 copyProperties 将数据拷贝给 Employee。然后再设置 EmployeeDTO 没有的属性即可。

(1) copyProperties 将一个类的属性值拷贝到另一个类上,但是一定要满足这个属性是两个类共有的。

(2)使用 MD5 给输入的密码加密,保证账号的安全性。

@Overridepublic void save(EmployeeDTO employeeDTO) {// 参数类型向员工类型转化Employee employee = new Employee();BeanUtils.copyProperties(employeeDTO, employee);// 设置账号状态employee.setStatus(StatusConstant.ENABLE);// 更新创建与修改信息时间employee.setCreateTime(LocalDateTime.now());employee.setUpdateTime(LocalDateTime.now());// 设置密文密码employee.setPassword(DigestUtils.md5DigestAsHex(PasswordConstant.DEFAULT_PASSWORD.getBytes()));// 隔离线程中获取数据Long  userId = BaseContext.getCurrentId();// 设置创建人和修改人employee.setCreateUser(userId);employee.setUpdateUser(userId);employeeMapper.insert(employee);}

隔离线程 ThreadLocal

        在新增员工列表中需要设置创建人与修改人,毫无疑问就是获取当前用户的信息,那么该怎么获取呢?我们可以使用隔离线程 -- ThreadLocal。

        ThreadLocal,也称为线程局部变量,是一种特殊的变量。它的特点是,每个线程都有该变量的一个副本,线程之间互不影响,实现了线程间的数据隔离。

        简单来讲就是客户端为每位用户都提供了单独的线程,而每个线程在 ThreadLocal 设置存取的值都是相互独立的。

这里可以简单的验证一下: 

        我们在令牌验证、controller、service 的地方加上一下代码,判断当前线程是否相同:

我们发现是相同的,那么就可以利用这一条特性,获取当前用户的信息:

        我们可以在获取令牌这里将解析的用户id存入 ThreadLocal 线程中,并在设置创建人或者修改人的时候取出,这样就可以获取当前的用户信息了。

Long  userId = BaseContext.getCurrentId();

Mapper

    @Insert("insert into employee(name, username, password, phone, sex, id_number, status, create_time, update_time, create_user, update_user) " +"VALUES (#{name},#{username},#{password},#{phone},#{sex},#{idNumber},#{status},#{createTime},#{updateTime},#{createUser},#{updateUser})")void insert(Employee Employee);

全局异常处理器

         因为我们数据库表的设计中,用户名是唯一属性,那么再次添加这个用户名的时候就会抛出500异常:

 这个异常是一个 SQL 异常是因为 username 冲突造成的 "Duplicate entry..."

        那么我们肯定不能直接将这个异常返回给用户,必须对它进行一定的处理,这个时候就需要用到全局异常处理器:

为什么需要全局异常处理器

不用强制写 try-catch,由全局异常处理器统一捕获处理。
自定义异常,只能用全局异常来捕获。不能直接返回给客户端,客户端是看不懂的,需要接入全局异常处理器。

处理 SQL 异常 

    @ExceptionHandlerpublic Result exceptionHandler(SQLIntegrityConstraintViolationException ex){// Duplicate entry '123' for key 'idx_username'String msg = ex.getMessage();if(msg.contains("Duplicate entry")){String[] split = msg.split(" ");String key = split[2];return Result.error(key+ MessageConstant.ALREADY_EXISTS);}return Result.error(MessageConstant.UNKNOWN_ERROR);}

        利用 ex.getMessage() 来获取报错信息,如果是 Duplicate entry 开头的异常信息那么铁定是 username 冲突问题,那么我们就需要提取 username 信息。并返回我们自定义的报错信息。如果开头不是 Duplicate entry 那么久返回未知异常。

        Result.error() 返回的错误信息不要写死,要同一进行管理,否则以后项目写大了需要更改需求就十分难找。

        我们可以采用静态常量的属性类来解决这个问题。

        这样我们的控制台就不会再抛出异常,前端也会接收到 username 已存在的信息。

代码测试

 

员工分页查询


 分页查询接口设计

 分析需求:

<1>员工信息分页查询后端返回的对象类型为:Result<PageResult>

<2>其实任何分页查询的底层都是加入了 limit 关键字分页

<3>为了内部数据模型与外部参数解耦,我们依旧采用 DTO 的设计模式

mybatis 提供的分页查询框架 pagehelper 依赖:

            <dependency><groupId>com.github.pagehelper</groupId><artifactId>pagehelper-spring-boot-starter</artifactId><version>1.3.0</version></dependency>

 Controller

    @GetMapping("/page")@ApiOperation("员工分页查询")public Result<PageResult> pageQuery(EmployeePageQueryDTO employeePageQueryDTO){log.info("员工分页查询:{}",employeePageQueryDTO);PageResult pageResult = employeeService.pageQuery(employeePageQueryDTO);return Result.success(pageResult);}

Service

        分页的核心就一行代码, PageHelper.startPage(page,pageSize) 这个就表示开始分页。加了这个之后 pagehelper 插件就会通过其内部的拦截器,将执行的 sql 语句,转化为分页的 sql 语句。

    @Overridepublic PageResult pageQuery(EmployeePageQueryDTO employeePageQueryDTO) {//开启分页查询PageHelper.startPage(employeePageQueryDTO.getPage(),employeePageQueryDTO.getPageSize());Page<Employee> page =  employeeMapper.pageQuery(employeePageQueryDTO);long total = page.getTotal();List<Employee> employees = page.getResult();return new PageResult(total,employees);}

 Mapper

使用动态SQL对name进行模糊查询,并以创建时间排降序。

    <select id="pageQuery" resultType="com.sky.entity.Employee">SELECT * FROM employee<where><if test="name!=null and name!=''">and name like concat('%',#{name},'%')</if></where>ORDER BY create_time DESC</select>

代码测试

 


http://www.ppmy.cn/embedded/138907.html

相关文章

docker 部署freeswitch(非编译方式)

一&#xff1a;安装部署 1.拉取镜像 参考&#xff1a;https://hub.docker.com/r/safarov/freeswitch docker pull safarov/freeswitch 2.启动镜像 docker run --nethost --name freeswitch \-e SOUND_RATES8000:16000 \-e SOUND_TYPESmusic:en-us-callie \-v /home/xx/f…

51单片机基础01 单片机最小系统

目录 一、什么是51单片机 二、51单片机的引脚介绍 1、VCC GND 2、XTAL1 2 3、RST 4、EA 5、PSEN 6、ALE 7、RXD、TXD 8、INT0、INT1 9、T0、T1 10、MOSI、MISO、SCK 11、WR、RD 12、通用IO P0 13、通用IO P1 14、通用IO P2 三、51单片机的最小系统 1、供电与…

小林Coding—Java「五、Java虚拟机面试篇」

五、Java虚拟机面试篇&#xff08;难⭐️⭐️⭐️&#xff09; 内存模型 JVM的内存模型介绍一下 JVM运行时内存共分为虚拟机栈、堆、元空间、程序计数器、本地方法栈 五个部分。还有一部分内存叫直接内存&#xff0c;属于操作系统的本地内存&#xff0c;也是可以直接操作的。 …

thinkphp6 入门(2)--视图、渲染html页面、赋值

use think\facade\View;View::assign([name > ThinkPHP,email > thinkphpqq.com]);View::assign(data,[name > ThinkPHP,email > thinkphpqq.com]); View::fetch(index);助手函数 view(index, [name > ThinkPHP,email > thinkphpqq.com ]); 模板输出 {$na…

集群聊天服务器(12)nginx负载均衡器

目录 负载均衡器nginx负载均衡器优势 如何解决集群聊天服务器跨服务器通信问题&#xff1f;nginx的TCP负载均衡配置nginx配置 负载均衡器 目前最多只能支持2w台客户机进行同时聊天 所以要引入集群&#xff0c;多服务器。 但是客户连哪一台服务器呢&#xff1f;客户并不知道哪一…

【jvm】方法区的理解

目录 1. 说明2. 方法区的演进3. 内部结构4. 作用5.内存管理 1. 说明 1.方法区用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码缓存等数据。它是各个线程共享的内存区域。2.尽管《Java虚拟机规范》中把方法区描述为堆的一个逻辑部分&#xff0c;但它却…

泷羽sec渗透DC靶场(1)完全保姆级学习笔记

前言 本次学习的是在b站up主泷羽sec课程完整版跳转链接有感而发&#xff0c;如涉及侵权马上删除文章。 笔记的只是方便各位师傅学习知识&#xff0c;以下网站只涉及学习内容&#xff0c;其他的都与本人无关&#xff0c;切莫逾越法律红线&#xff0c;否则后果自负。 &#xff0…

SQLite3 JDBC Java工具类

最近生产环境mysql各种无法使用&#xff0c;要求下线。有一堆小工具&#xff0c;平时因为mysql用着方便&#xff0c;配置啊&#xff0c;临时的一些比对数据存在里面。迁移很麻烦。 发现SQLite 3.47.0版本之后&#xff0c;性能大增&#xff0c;支持多线程&#xff0c;记录级锁。…