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

Java 开发——(下篇)从零开始搭建后端基础项目 Spring Boot 3 + MybatisPlus

上篇速递 - Spring Boot 3 + MybatisPlus


五、静态资源访问

1. 基础配置

在 Spring Boot 中访问静态资源非常方便。Spring Boot 默认支持从以下位置加载静态资源

  • /META-INF/resources/
  • /resources/
  • /static/
  • /public/

这些目录下的文件可以直接通过 URL 访问。

例如,如果有一个图片文件放在 src/main/resources/static/logo.png,那么可以通过 http://localhost:8080/logo.png 来访问这个图片。(假设服务器运行在 8080 端口)


定义过滤规则

在默认的情况下,访问静态资源的 URLhttp://localhost:8080/图片名.文件后缀

如果想要自定义其他的过滤规则,可以在 application.properties 配置文件中进行配置。

例如,在 application.properties 中添加如下配置:

# 定义过滤规则 /images/**
spring.mvc.static-path-pattern=/images/**

这会告诉 Spring Boot 访问静态资源的 URLhttp://localhost:8080/images/图片名.文件后缀


配置静态资源路径

如果想自定义静态资源的位置,或添加额外的静态资源目录,可以在 application.properties 配置文件中进行配置。

例如,在 application.properties 中添加如下配置:

# 定义多个位置,使用逗号分隔
spring.resources.static-locations=classpath:/my-resources/,classpath:/static/

这会告诉 Spring Boot 从 /my-resources//static/ 这两个目录下加载静态资源。


2. 项目搭建

我们可以在项目中进行如下配置:

# 静态资源配置
# 定义过滤规则 /images/**
spring.mvc.static-path-pattern=/images/**

# 定义静态资源位置,对静态资源进行放行允许浏览器访问 classpath:/static/
#spring.web.resources.static-locations=classpath:/static/

# 定义静态资源位置,会覆盖原先默认的静态资源位置
# 定义多个位置,使用逗号分隔
# 用户上传文件位置 upload,相对于配置文件的位置,"/"代表服务器所在路径
spring.web.resources.static-locations=/upload/,classpath:/static/

3. 注意事项

当我们想要测试这个静态资源的访问时,可能会遇到以下这个问题:

我们使用 IDEA 开发,就会直接把电脑上的测试图片直接复制到项目的 /resources/static 目录下,然后用 http://localhost:8080/images/test.jpg 进行测试,发现测试图片加载失败了

这并非项目配置的问题。

而是由于导入的图片是放到项目的 static 目录下,而编译后的 target 中没有该资源,所以访问不到。

简单来说就是项目需要重新编译 Rebuild Project,重新编译后静态资源会被加载到 target 目录下,此时就可以通过浏览器访问到图片了。


六、文件上传

前端通过 API 接口上传文件是一个常见的需求,通常涉及以下几个步骤:前端准备文件数据、发送 HTTP 请求到后端、后端接收并处理文件。

在后端 Spring Boot 项目中,我们可以对接收的文件进行如下处理。

1. 配置文件上传

application.properties 中配置文件上传的最大大小:

# tomcat 限制请求上传文件的大小
# 每个文件配置最大大小,单个上传文件最大为 10MB
spring.servlet.multipart.max-file-size=10MB

# 单次请求的文件总大小
#spring.servlet.multipart.max-request-size=10MB

2. 创建控制器

创建一个控制器 FileUploadController 来处理文件上传请求:

@RestController
public class FileUploadController {

    @PostMapping("/upload")
    public String upload(String nickname, MultipartFile photo, HttpServletRequest request) throws IOException {
        System.out.println(nickname);

        //获取图片原始名称
        System.out.println(photo.getOriginalFilename());

        // 获取文件类型
        System.out.println(photo.getContentType());

        // 设置上传图片的保存位置
        // 这表示上传文件的位置是动态的,是 tomcat 服务器所在位置
        // 获取服务器上某个虚拟路径的实际物理路径
        String path = request.getServletContext().getRealPath("/upload/");
        
        // 打印出图片上传位置
        System.out.println(path);

        // 保存图片
        saveFile(photo, path);

        return "上传成功";
    }

    private void saveFile(MultipartFile photo, String path) throws IOException {

        // 判断存储目录是否存在,如果不存在则创建该目录
        File dir = new File(path);
        if (!dir.exists()) {
            // 创建目录
            dir.mkdir();
        }

        // 储存后文件
        File file = new File(path + photo.getOriginalFilename());
        // 把网络上的文件存储到服务器本地
        photo.transferTo(file);

    }

}

3. 设置静态资源放行

通过 request.getServletContext().getRealPath() 可以获取服务器上某个虚拟路径的实际物理路径,及 tomcat 服务器所在位置。

由上面的文件上传控制器所知,前端上传的图片被存放到了 tomcat 服务器所在位置的 upload 文件夹下。

如果我们想在前端(浏览器)访问到这些图片,可以把这个文件夹设置成静态资源目录

在配置文件 application.properties 中如何获取到这个目录?

在配置文件中 “/” 代表服务器所在路径,故这个目录用 “/upload/” 表示。

# 定义静态资源位置,会覆盖原先默认的静态资源位置
# 定义多个位置,使用逗号分隔
# 用户上传文件位置 upload,相对于配置文件的位置,"/"代表服务器所在路径
spring.web.resources.static-locations=/upload/,classpath:/static/

需要注意的点:在开发阶段,Spring Boot 项目的内置 tomcat 服务器所在位置不是固定的每次重启项目位置都会发生变化


4. 测试方法

使用 Postman 测试下这个接口,文件上传的格式一般选择 form-data 格式。

在这里插入图片描述


在浏览器中访问:

http://localhost:8080/images/bus.jpg

七、拦截器

在 Spring Boot 中,拦截器(Interceptor)是一种常用的机制,用于在请求到达控制器之前或之后执行某些操作。拦截器可以用于日志记录、权限校验、性能监控等多种场景。

下面是如何在 Spring Boot 中定义和使用拦截器的详细步骤。

1. 定义拦截器

创建拦截器类:实现 HandlerInterceptor 接口或继承 HandlerInterceptorAdapter 类。

如,在项目中我们创建了一个拦截器类 LoginInterceptor,让它继承 HandlerInterceptor 接口。

返回值 return 表示是否拦截:

  • true:不拦截;
  • false:拦截。
// 添加拦截器
public class LoginInterceptor implements HandlerInterceptor {

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        // 拦截器打印的信息,在控制台看到它就说明拦截器生效了
        System.out.println("拦截器打印的信息");
        return true;
        // 如果 false 代表拦截,具体的拦截规则在拦截器注册文件中
    }
}

// config 中注册拦截器,使拦截器生效

2. 注册拦截器

我们需要在配置类中注册拦截器,拦截器才会生效。

注册拦截器:创建一个配置类,实现 WebMvcConfigurer 接口,并重写 addInterceptors 方法。

如,在项目中我们创建了配置类 WebConfig,让它继承 WebMvcConfigurer 接口,重写 addInterceptors 方法,同时添加注解 @Configuration 使配置类生效。

addInterceptors 方法中,可以设置拦截的具体位置。

// 添加注解 @Configuration 使配置类生效
@Configuration
public class WebConfig implements WebMvcConfigurer {

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        // config 中注册拦截器,使拦截器生效
        
        // 表示拦截所有的请求
//        registry.addInterceptor(new LoginInterceptor());

        // 有针对性的拦截
        // /hello 表示路由映射的路径 http://localhost:8080/hello?nickname=zhangsan&phone=123
        // 拦截所有/hello/开头的请求
        registry.addInterceptor(new LoginInterceptor()).addPathPatterns("/hello/**");
    }
}

测试一个 /hello/ 开头的请求,观察后端控制台中拦截器打印的信息。


八、RESTful 风格

RESTful(Representational State Transfer)风格是一种设计网络应用程序架构的方式,它利用 HTTP 协议的特性来实现客户端和服务器之间的通信。

RESTful API 设计的核心原则是使用标准的 HTTP 方法(如 GET、POST、PUT、DELETE 等)来操作资源,每个资源都有唯一的标识符(通常是 URL)。

下面是一些关于如何设计 RESTful API 的基本指南。

1. RESTful API 设计原则

资源命名

  • 使用名词而不是动词来表示资源。
  • 使用复数形式来表示集合资源。
  • 使用连字符或驼峰命名法来提高可读性。

HTTP 方法(关键点)

  • GET:用于获取资源。
  • POST:用于创建资源。
  • PUT:用于更新资源(替换整个资源)。
  • PATCH:用于部分更新资源。
  • DELETE:用于删除资源。

状态码

  • 200 OK:请求成功。
  • 201 Created:资源已创建。
  • 204 No Content:请求成功,但没有返回内容。
  • 400 Bad Request:客户端请求有误。
  • 401 Unauthorized:请求未授权。
  • 403 Forbidden:请求被拒绝。
  • 404 Not Found:资源未找到。
  • 500 Internal Server Error:服务器内部错误。

版本控制

  • 可以通过 URL 路径或请求头来实现API版本控制。

2. 在项目中应用

实体类

需要用到我们之前创建的实体类 User

// 用于封装参数的实体类
public class User {

    private String username;
    private String password;

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    @Override
    public String toString() {
        return "User{" +
                "username='" + username + '\'' +
                ", password='" + password + '\'' +
                '}';
    }
}

控制类

创建一个控制器类 UserController 来尝试使用这个 RESTful 风格。

@RestController
public class UserController {

    // Get:用于获取资源
    // 动态方式获取 {id},需要添加注解 @PathVariable
    @GetMapping("/user/{id}")
    public String getUserById(@PathVariable int id) {
        return "根据 ID 获取用户信息:" + id;
    }

    // Post:用于创建资源
    @PostMapping("/user")
    public String addUser(User user) {
        System.out.println(user.toString());
        return "添加用户";
    }

    // Put:用于更新资源
    @PutMapping("/user")
    public String updateUser(User user) {
        return "更新用户";
    }

    // Delete:用于删除资源
    @DeleteMapping("/user/{id}")
    public String deleteUser(@PathVariable int id) {
        return "根据 ID 删除用户信息:" + id;
    }

}

测试方法

比如测试一下第一个 GET 方法。

在这里插入图片描述


九、Swagger3

在 Spring Boot 3.0 中使用 Swagger 所需要的依赖和 2.7 有所不同。

原因很简单,Spring Boot 3.0 是一次大版本升级,故造成了很多不兼容更新。

例如最低兼容的 Java 版本为 17,底层也切换到了 Spring 6。

所以自 2020 年以来没有更新过的 springfox-swagger2 和 springfox-swagger-ui 自然不会兼容 2022 年发布的 Spring Boot 3.0。


1. 添加 Maven 依赖

pom.xml 文件中添加 Maven 依赖,用于 Swagger3 的使用。

<!-- Swagger(openAPI) 相关功能依赖-->
<dependency>
    <groupId>org.springdoc</groupId>
    <artifactId>springdoc-openapi-starter-webmvc-ui</artifactId>
    <version>2.6.0</version>
</dependency>
<!--  搭配校验使用,使用与 Spring Boot 相同的版本号  -->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-validation</artifactId>
    <version>3.3.4</version>
</dependency>

2. 配置 Swagger3

application.properties 中添加如下配置,启用并配置 OpenAPISwagger UI

OpenAPISwagger UI 有自己默认的访问地址,当然也可以自己设置。

# 配置 swagger
# 启用并配置 OpenAPI 文档生成
springdoc.api-docs.enabled=true
# 默认地址 http://localhost:8080/v3/api-docs
# 自定义访问地址
#springdoc.api-docs.path=/user-service/v3/api-docs

# 配置 Swagger UI
springdoc.swagger-ui.enabled=true
# 默认地址 http://localhost:8080/swagger-ui/index.html
# 自定义访问地址
#springdoc.swagger-ui.path=/user-service/swagger-ui/index.html

然后,创建 Swagger3 的配置类 SwaggerConfig,对文档的一些细节进行设置。

  • title - 标题
  • description - 描述
  • version - 版本
@Configuration
public class SwaggerConfig {
    @Bean
    public OpenAPI swaggerOpenApi() {
        return new OpenAPI()
                .info(new Info().title("XXX微服务平台")
                        .description("描述平台")
                        .version("v1.0.0"))
                .externalDocs(new ExternalDocumentation()
                        .description("设计文档")
                        .url("https://blog.csdn.net/Sareur_1879?type=blog"));
    }
}

3. 网址访问

然后启动项目,就可以通过网址进行访问 API 文档了。

// 默认地址 http://localhost:8080/swagger-ui/index.html
// 自定义访问地址 http://localhost:8080/user-service/swagger-ui/index.html

4. 存在的问题

在 Swagger2 版本是可以正常上传文件的(我们上面那个文件上传接口是可以正常使用的,文档中的参数类型 form-data 是对的)

但是,新版本的 OpenAPI(Swagger3) 在文件上传方面存在问题,在 Postman 中测试文件上传功能正确后,Swagger3 中无法正确识别上传参数的类型 form-data,它显示的是 String

这导致了文件类型上传不成功。(Postman 测试是成功的,Swagger3 反而不成功,可见 Swagger3 也不是很可信)

这个问题笔者尝试了多种方法后尚未解决,期待好心人的告知。


十、MyBatisPlus

MyBatisPlus 是一个 MyBatis 的增强工具,旨在简化开发、提高效率。它在 MyBatis 的基础上只做增强不做改变,为简化开发、提高效率而生。接下来我们在 Spring Boot 项目中尝试使用 MyBatisPlus。

1. 添加 Maven 依赖

pom.xml 文件中添加。

Lombok

Lombok 是一个 Java 库,使用 Lombok 可以简化实体类的代码,例如自动生成 gettersetter 方法、构造函数、toString 方法等,可以减少样板代码的编写。

<!-- lombok 用于自动 getter/setter -->
<dependency>
    <groupId>org.projectlombok</groupId>
    <artifactId>lombok</artifactId>
    <version>1.18.34</version>
    <scope>provided</scope>
</dependency>

MyBatisPlus 需要的依赖

<!-- MyBatisPlus-->
<dependency>
    <groupId>com.baomidou</groupId>
    <artifactId>mybatis-plus-spring-boot3-starter</artifactId>
    <version>3.5.8</version>
</dependency>

<!-- MySQL 驱动-->
<dependency>
    <groupId>mysql</groupId>
    <artifactId>mysql-connector-java</artifactId>
    <version>5.1.49</version>
</dependency>

<!-- 数据连接池 druid -->
<dependency>
    <groupId>com.alibaba</groupId>
    <artifactId>druid-spring-boot-starter</artifactId>
    <version>1.2.23</version>
</dependency>

2. ORM 框架

ORM(Object-Relational Mapping,对象关系映射)是一种编程技术,用于将对象模型与关系型数据库模型之间进行转换。简单来说,ORM 允许开发者使用面向对象的编程语言(如 Java)来操作数据库,而不需要直接编写 SQL 语句。这种方式可以提高开发效率,减少错误,并且使代码更易于维护。

MyBatis 是一个非常强大且灵活的 ORM 框架,特别适合需要精细控制 SQL 语句的场景。


配置数据库

在配置文件 application.properties 中配置数据库

# 数据库配置
spring.datasource.type=com.alibaba.druid.pool.DruidDataSource
# 用 jdbc 连接数据库
spring.datasource.driver-class-name=com.mysql.jdbc.Driver
# mydb 对应本地 mysql 的数据库
spring.datasource.url=jdbc:mysql://localhost:3306/mydb?useSSL=false
spring.datasource.username=数据库账号
spring.datasource.password=数据库密码
mybatis-plus.configuration.log-impl=org.apache.ibatis.logging.stdout.StdOutImpl

创建数据表

使用 Navicat 工具连接本地 MySQL 数据库,新建数据库 mydb,在 mydb 中创建数据表 user

表结构如下,可以新建查询,运行 sql 文件直接生成。

-- ----------------------------
-- Table structure for user
-- ----------------------------
DROP TABLE IF EXISTS `user`;
CREATE TABLE `user`  (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `username` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
  `password` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
  `birthday` varchar(255) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
  PRIMARY KEY (`id`) USING BTREE
) ENGINE = InnoDB AUTO_INCREMENT = 6 CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = Dynamic;

SET FOREIGN_KEY_CHECKS = 1;

然后插入一些测试数据

-- ----------------------------
-- Records of user
-- ----------------------------
INSERT INTO `user` VALUES (1, 'zhangsan', '123456', '2020-01-13');
INSERT INTO `user` VALUES (2, 'lisi', '123', '2020-03-13');
INSERT INTO `user` VALUES (3, 'wangwu', '14567', '2023-07-09');

在 Navicat 中新建查询,选择对应的数据库,写入 sql 语句,然后点击运行。

在这里插入图片描述


修改实体类

@Data 是 Lombok 提供的一个非常强大的注解,它可以一次性给类生成多个常用的方法,从而大大减少样板代码的编写。具体来说,@Data 注解会为类中的每个字段自动生成以下这些方法:

  • Getter 方法:为每个字段生成 get 方法。
  • Setter 方法:为每个字段生成 set 方法。
  • toString 方法:生成一个 toString 方法,包含所有字段的信息。
  • equals 和 hashCode 方法:生成 equalshashCode 方法,用于比较对象是否相等。
  • RequiredArgsConstructor:生成一个包含所有 final 字段和 @NonNull 字段的构造函数。

实体类 User 对应数据库 MySQL 中的 user 表,user 表中的字段与 User 中的变量一一对应。

  • 表名与类名应该一致;
  • 如果表名与类名不一致,使用 @TableName 注解进行匹配;
  • @TableId 注解用于标记实体类中的主键字段,如 id;
  • IdType.AUTO 表示自增。

配合添加的 Lombok 使用,我们修改一下 User 类,使用 @Data 注解后,实体类中的样板代码明显减少。

/**
 * @TableName("user")
 * 如果表名与类名不一致,使用该注解进行匹配
 */
@Data   // Lombok 会自动生成相关 Getter/Setter 方法,以及 toString 方法
public class User {

    /**
     * @TableId 该注解用于标记实体类中的主键字段
     * IdType.AUTO 自增
     */
    @TableId(type = IdType.AUTO)
    private int id;
    private String username;
    private String password;
    private String birthday;

}

3. 修改主启动类

在主启动文件中添加 @MapperScan 注解,扫描 mapper 包。

UserMapper 接口就放在 mapper 包下面,记得要写一整串的包名,建议右键 Copy Reference

在这里插入图片描述


4. 创建 Mapper 接口

创建 UserMapper 接口

  • 添加注解 @Mapper
  • 继承 BaseMapper<User> 接口,根据传入的 User 自动匹配数据库中同名的表。
  • MyBatis-Plus 提供了丰富的 CRUD(增删改查) 操作。
  • 可以调用 BaseMapper 中写好的方法,实现数据增删改查。一些 SQL 语句不需要自己重复写了。
@Mapper
public interface UserMapper extends BaseMapper<User> {
    /**
     * 继承 BaseMapper<User>
     * 根据传入的 User 自动匹配数据库中同名的表
     * 可以调用 BaseMapper 中写好的方法,实现数据增删改查
     */

}

5. 创建 Controller 层

创建 UserController2:响应前端网络请求。

在项目中,我们使用 MybatisPlus 提供的两个操作:

  • 查询:调用 BaseMapper 中的 selectList() 方法,null 表示没有查询条件,也就是查询全部记录的意思。
  • 插入数据
    • 调用 BaseMapper 中的 insert() 方法,user 对象中的数据是前端发送过来的,需要插入到底层的数据库表中;
    • id 是自增字段,不需要插入,会自己生成;
    • 由于 User 类的 id 使用了注解标注为主键和自增,当我们在控制台打印 user 对象时,会自动读取数据表中的 id 字段。这就是为什么前端请求中没有参数 id,后端控制台却能打印出 id 的原因。
  • BaseMapper 是自带的,UserMapper 继承了这个接口,子类对象可以使用父类中的一些方法
@RestController
public class UserController2 {

    @Autowired  // spring 注入 UserMapper
    private UserMapper userMapper;

    // 查询
    @GetMapping("/user2")
    public List<User> query() {
        List<User> list = userMapper.selectList(null);
        System.out.println(list);
        return list;
    }

    // 插入数据
    @PostMapping("/user2")
    public String save(User user) {
        // 当数据插入成功后,insert 方法的返回值大于 0
        int i = userMapper.insert(user);
        // 虽然没有插入 id,但使用注解标注为主键和自增,user 中会读取数据表中的 id,然后打印到控制台上
        System.out.println(user);
        if (i > 0) {
            return "插入成功";
        } else {
            return "插入失败";
        }
    }
}

6. 测试方法

使用 Postman 工具对这两个 API 接口进行测试。

查找测试

在这里插入图片描述



新增测试

在这里插入图片描述


(全文完)


参考资料

上篇速递 - Spring Boot 3 + MybatisPlus:https://blog.csdn.net/Sareur_1879/article/details/143181790

参考学习视频 - SpringBoot + Vue 全栈开发:https://www.bilibili.com/video/BV1nV4y1s7ZN/

参考文章 - Java 开发环境搭建:https://blog.csdn.net/Sareur_1879/article/details/137963848

参考文章 - Maven 在 IDEA 中的配置与使用:https://blog.csdn.net/Sareur_1879/article/details/143091933

参考文章 - Spring Boot 项目是两种创建方式:https://blog.csdn.net/Sareur_1879/article/details/139201323

Maven 官网:https://maven.apache.org/download.cgi

MVN 仓库:https://mvnrepository.com/

Spring Boot 官网:https://spring.io/projects/spring-boot

Oracle 官网:https://www.oracle.com

IDEA 官网:https://www.jetbrains.com.cn/idea/

Postman 官网:https://www.postman.com/

Notepad++ 下载地址:https://notepad-plus.en.softonic.com/

WinRAR 下载地址:https://www.winrar.com.cn/


http://www.kler.cn/a/373593.html

相关文章:

  • Oracle FLOOR函数的用法
  • 3D目标检测数据集——Waymo数据集
  • 计算机网络 | 什么是公网、私网、NAT?
  • 解决win11的vmvare和docker冲突
  • Redis常见
  • 一路相伴,非凸科技助力第49届ICPC亚洲区决赛
  • C# .NET最小API?
  • 【利器】12个评估大语言模型(LLM)质量的自动化框架
  • GAME JAM:加入我们的甜蜜幽灵冒险之旅
  • Centos安装ffmpeg的方法
  • Electron调用nodejs的cpp .node扩展【非安全】
  • 「图文详解」Pycharm 远程服务器Debug
  • 【SSM-Day5】SpringMVC入门
  • Redis新数据类型
  • 基于vue框架的的驾校预约车辆管理系统设计与实现jwoqj(程序+源码+数据库+调试部署+开发环境)系统界面在最后面
  • [瑞吉外卖]-10前后端分离
  • Vue3学习:vue组件中的图片路径问题
  • 字符函数和字符串函数(C 语言)
  • 微服务电商平台课程二:技术图谱
  • AI产品独立开发变现实战营,炒掉老板做自由职业赚大钱
  • 【vue 全家桶】1、vue 基础
  • Pytorch学习--DataLoader的使用
  • 验证工程师如何使用UVM
  • springcloud网关和熔断配置
  • YOLOv11改进策略【SPPF】| NeuralPS-2022 Focal Modulation : 使用焦点调制模块优化空间金字塔池化SPPF
  • C++变量声明与定义(有对引用的解释)