(七)springboot实战——springboot3集成R2DBC实现webflux响应式编程服务案例

前言

本节主要内容是关于使用新版springboot3集成响应式数据库R2DBC,完成响应式web服务案例。需要注意的是,此次项目使用的JDK版本是JDK17,springboot版本使用3.2.2版本,数据库使用关系型数据库mysql。WebFlux 是一个基于响应式编程模型的框架,适用于构建异步、非阻塞的高性能 Web 应用程序。它具有高并发能力、函数式编程风格、与其他 Spring 框架的集成能力等优势,可以满足对性能要求高或需要处理大量并发请求的场景。

WebFlux 使用响应式编程模型,基于 Reactor 库实现。它利用异步和非阻塞的方式处理请求,能够处理大量的并发请求,提高应用程序的性能和可扩展性。由于采用了非阻塞的 I/O 操作,WebFlux 能够更有效地利用服务器资源,处理更多的并发请求,提供更高的吞吐量。WebFlux 鼓励使用函数式和声明式的编程风格,通过组合一系列操作来处理请求和响应。这种编程方式更具可读性和可维护性。WebFlux 使用 Flux 和 Mono 这两个反应式类型来处理异步数据流。Flux 用于表示包含零个或多个元素的异步序列,而 Mono 用于表示包含零个或一个元素的异步结果。WebFlux 可以与其他 Spring 框架和库进行无缝集成,例如 Spring Boot、Spring Data、Spring Security 等。这使得开发者可以充分利用 Spring 生态系统的优势。WebFlux 不依赖于特定的服务器,可以在各种服务器上运行,包括 Netty、Undertow 和 Tomcat 等。这为开发者提供了更多的灵活性和选择性。

正文

①使用idea创建一个springboot3项目

②引入r2dbc的mysql驱动和响应式Spring Data R2dbc启动器以及接口文档springdoc

<dependency>
     <groupId>org.projectlombok</groupId>
     <artifactId>lombok</artifactId>
</dependency>

<!-- https://mvnrepository.com/artifact/org.springdoc/springdoc-openapi-starter-webflux-ui -->
<dependency>
	<groupId>org.springdoc</groupId>
	<artifactId>springdoc-openapi-starter-webflux-ui</artifactId>
	<version>2.3.0</version>
</dependency>

<!--        响应式 Spring Data R2dbc-->
<dependency>
	<groupId>org.springframework.boot</groupId>
	<artifactId>spring-boot-starter-data-r2dbc</artifactId>
</dependency>

<!-- https://mvnrepository.com/artifact/io.asyncer/r2dbc-mysql -->
<dependency>
	<groupId>io.asyncer</groupId>
	<artifactId>r2dbc-mysql</artifactId>
	<version>1.0.6</version>
</dependency>

 ③在resource目录下创建配置文件application.yaml,配置mysql的连接配置信息和接口文档

spring:
  r2dbc:
    url: r2dbc:mysql://192.168.110.88:3306/ht-atp
    username: root
    password: root
server:
  port: 8080
springdoc:
  api-docs:
    enabled: true
    path: /api-docs
  swagger-ui:
    enabled: true
    path: /swagger-ui/index.html
  show-actuator: true

 ④启动项目,验证项目是否可以正常启动,以及接口文档是否可以正常使用

⑤创建用户实体类User

package com.yundi.atp.entity;

import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Data;
import org.springframework.data.annotation.Id;
import org.springframework.data.relational.core.mapping.Table;


@Schema(name = "User", description = "用户类")
@Table(name = "user")
@Data
public class User {
    @Id
    @Schema(name = "id", description = "用户ID")
    private Integer id;

    @Schema(name = "name", description = "用户名称")
    private String name;

    @Schema(name = "age", description = "用户年龄")
    private Integer age;
}

⑥创建统一接口响应对象ApiResponse

package com.yundi.atp.common;

import io.swagger.v3.oas.annotations.media.Schema;
import lombok.Builder;
import lombok.Data;


@Data
@Builder
public class ApiResponse<T> {
    @Schema(name = "code", description = "状态码")
    private Integer code;

    @Schema(name = "msg", description = "消息结果")
    private String msg;

    @Schema(name = "data", description = "数据")
    private T data;

    /**
     * 成功统一响应格式
     *
     * @param
     * @param <T>
     * @return
     */
    public static <T> ApiResponse<T> success() {
        ApiResponse<T> apiResponse = new ApiResponse<>(200, "成功", null);
        return apiResponse;
    }

    /**
     * 成功统一响应格式
     *
     * @param data
     * @param <T>
     * @return
     */
    public static <T> ApiResponse<T> success(T data) {
        ApiResponse<T> apiResponse = new ApiResponse<>(200, "成功", data);
        return apiResponse;
    }

    /**
     * 失败统一响应格式
     *
     * @param code
     * @param message
     * @param <T>
     * @return
     */
    public static <T> ApiResponse<T> fail(Integer code, String message) {
        return new ApiResponse<>(code, message, null);
    }
}

 ⑦创建UserController控制类,完成用户管理接口申明

package com.yundi.atp.controller;

import com.yundi.atp.common.ApiResponse;
import com.yundi.atp.entity.User;
import com.yundi.atp.service.UserService;
import io.swagger.v3.oas.annotations.Operation;
import io.swagger.v3.oas.annotations.tags.Tag;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;

import java.util.List;


@Tag(name = "用户管理", description = "用户管理")
@RestController
@RequestMapping("/user")
public class UserController {
    @Autowired
    private UserService userService;

    @Operation(summary = "获取全部用户信息", description = "获取全部用户信息")
    @GetMapping(value = "getAllUserInfo")
    public Mono<ApiResponse<List<User>>> getAllUserInfo() {
        Flux<User> userInfo = userService.getAllUserInfo();
        return userInfo.collectList().map(user -> ApiResponse.success(user));
    }

    @Operation(summary = "根据ID获取用户信息", description = "根据ID获取用户信息")
    @GetMapping(value = "getUserById/{id}")
    public Mono<ApiResponse<User>> getUserById(@PathVariable("id") Integer id) {
        Mono<User> user = userService.getUserById(id);
        Mono<ApiResponse<User>> userMono = user.map(ApiResponse::success);
        return userMono;
    }

    @Operation(summary = "新增用户", description = "新增用户")
    @PostMapping(value = "saveUser")
    public Mono<ApiResponse> saveUser(@RequestBody User user) {
        Mono<User> userMono = userService.saveUser(user);
        return userMono.thenReturn(ApiResponse.success());
    }

    @Operation(summary = "更新用户", description = "更新用户")
    @PostMapping(value = "updateUser")
    public Mono<ApiResponse> updateUser(@RequestBody User user) {
        Mono<User> userMono = userService.updateUser(user);
        return userMono.thenReturn(ApiResponse.success());
    }

    @Operation(summary = "删除用户", description = "删除用户")
    @DeleteMapping(value = "deleteUser/{id}")
    public Mono<ApiResponse> deleteUser(@PathVariable("id") Integer id) {
        Mono<Void> userMono = userService.deleteUser(id);
        return userMono.thenReturn(ApiResponse.success());
    }
}

⑧ 创建用户管理接口层UserService,定义接口

package com.yundi.atp.service;

import com.yundi.atp.entity.User;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;

public interface UserService {
    /**
     * 查询全部用户信息
     *
     * @return
     */
    Flux<User> getAllUserInfo();

    /**
     * 根据ID获取用户信息
     *
     * @param id
     * @return
     */
    Mono<User> getUserById(Integer id);

    /**
     * 新增用户
     *
     * @param user
     * @return
     */
    Mono<User> saveUser(User user);

    /**
     * 更新用户
     *
     * @param user
     * @return
     */
    Mono<User> updateUser(User user);

    /**
     * 删除用户
     *
     * @param id
     * @return
     */
    Mono<Void> deleteUser(Integer id);
}

 ⑨创建用户管理实现层UserServiceImpl,完成用户管理的业务逻辑

package com.yundi.atp.service.impl;

import com.yundi.atp.entity.User;
import com.yundi.atp.repository.UserRepository;
import com.yundi.atp.service.UserService;
import jakarta.annotation.Resource;
import org.springframework.stereotype.Service;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;

@Service
public class UserServiceImpl implements UserService {
    @Resource
    private UserRepository userRepository;

    @Override
    public Flux<User> getAllUserInfo() {
        return userRepository.findAll();
    }

    @Override
    public Mono<User> getUserById(Integer id) {
        return userRepository.findById(id);
    }

    @Override
    public Mono<User> saveUser(User user) {
        return userRepository.save(user);
    }

    @Override
    public Mono<User> updateUser(User user) {
        return userRepository.updateUser(user);
    }

    @Override
    public Mono<Void> deleteUser(Integer id) {
        return userRepository.deleteById(id);
    }
}

⑩ 创建UserRepository持久层,并继承R2dbcRepository通用数据库操作接口完成数据库的操作

package com.yundi.atp.repository;

import com.yundi.atp.entity.User;
import org.springframework.data.r2dbc.repository.Query;
import org.springframework.data.r2dbc.repository.R2dbcRepository;
import org.springframework.data.repository.query.Param;
import org.springframework.stereotype.Repository;
import reactor.core.publisher.Mono;


@Repository
public interface UserRepository extends R2dbcRepository<User, Integer> {
    /**
     * 更新用户信息
     *
     * @param user
     * @return
     */
    @Query("update user set name = :#{#user.name}, age = :#{#user.age} where id = :#{#user.id}")
    Mono<User> updateUser(@Param("user") User user);


    /**
     * 动态更新用户信息
     *
     * @param user
     * @return
     */
    @Query("UPDATE User SET " +
            "name = CASE WHEN :#{#user.name} IS NULL THEN name ELSE :#{#user.name} END, " +
            "age = CASE WHEN :#{#user.age} IS NULL THEN age ELSE :#{#user.age} END " +
            "WHERE id = :#{#user.id}")
    Mono<User> updateUserQuery(@Param("user") User user);
}

⑪ 使用swagger接口文档测试用户管理接口

结语

至此,关于springboot3集成R2DBC实现webflux响应式编程服务案例的内容到这里就结束了,我们下期见。。。。。。

相关推荐

  1. 响应编程——R2DBC

    2024-01-27 16:34:03       36 阅读
  2. 响应编程WebFlux基础实战练习

    2024-01-27 16:34:03       29 阅读
  3. SpringBoot整合 R2DBC

    2024-01-27 16:34:03       18 阅读
  4. Vue3实现响应编程

    2024-01-27 16:34:03       28 阅读

最近更新

  1. TCP协议是安全的吗?

    2024-01-27 16:34:03       16 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2024-01-27 16:34:03       16 阅读
  3. 【Python教程】压缩PDF文件大小

    2024-01-27 16:34:03       15 阅读
  4. 通过文章id递归查询所有评论(xml)

    2024-01-27 16:34:03       18 阅读

热门阅读

  1. Linux命令-apk命令(Alpine Linux 下的包管理工具)

    2024-01-27 16:34:03       30 阅读
  2. 树莓派开机重启测试

    2024-01-27 16:34:03       34 阅读
  3. C语言什么是运算符的目?怎样进行区分?

    2024-01-27 16:34:03       34 阅读
  4. 【mysql把一个字段分割成两个字段】

    2024-01-27 16:34:03       35 阅读
  5. 考研机试 成绩排序

    2024-01-27 16:34:03       27 阅读
  6. 七、SQL编程

    2024-01-27 16:34:03       25 阅读
  7. vue2后台管理项目权限的分类

    2024-01-27 16:34:03       29 阅读
  8. 系统架构16 - 软件工程(4)

    2024-01-27 16:34:03       27 阅读
  9. Element修改树结构样式--虚线树

    2024-01-27 16:34:03       27 阅读
  10. 有关递推题目的感想(继上篇文章)

    2024-01-27 16:34:03       29 阅读
  11. SQL Server 中,删除表数据有以下几种方式

    2024-01-27 16:34:03       37 阅读
  12. 代码随想录算法训练营数组总结

    2024-01-27 16:34:03       44 阅读
  13. ubuntu 22.04 安装redis并设置远程连接

    2024-01-27 16:34:03       38 阅读