SpringBoot+Mybatis-Plus实现增删改查超详细步骤

这篇具有很好参考价值的文章主要介绍了SpringBoot+Mybatis-Plus实现增删改查超详细步骤。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

目录

一、介绍

视频讲解

二、前期准备工作

(一) 创建springboot项目和创建数据库

三、项目配置

(一)pom.xl导入相关依赖 

1.导入依赖

(二)yml文件中配置连接数据库

2.配置yml文件 

四、代码的编写

数据库展示

项目提前展示!!!

(三)MySQL表绑定到spring boot(实体层)

3.1 创建实体包

 3.2编写User类

(四)springboot绑定到mybatis-plus(数据层)

4.1创建mapper包

4.2编写UserMapper接口

(五)把数据层加工处理成逻辑业务(service服务层)

5.1创建service包 

5.2编写获取内置增删改查方法接口IUserService

5.3编写自己定义的UserService

5.4编写服务层合成逻辑业务UserServiceImpl

5.5故事解读服务层接口和类

(六)调用服务层的皇帝

 6.1创建controller包

6.2 编写调取业务层的UserController类

五、前端请求测试


一、介绍

本篇文章通过springboot整合mybatis-plus去实现后端对数据库的增删改查以及响应给前端的url,让前端获得数据。mybatis-plus技术是简化了繁琐的代码操作,把增删改查的语句都内置了,直接调用就可以实现数据库的增删改查了。还可以一定程度上防止SQL注入。缺点就是不够灵活,不过这个灵活的问题也可以自己去创建代码去弥补这个。

视频讲解

springboot整合mybatis-plus增删改查快速上手_哔哩哔哩_bilibili

二、前期准备工作

(一) 创建springboot项目和创建数据库

1. spring boot项目的介绍和创建方式:SpringBoot项目的快速创建方式(包含第一个程序的运行)_云边的快乐猫的博客-CSDN博​​​​​​客

2.创建MySQL数据库和新建表的详细步骤:

 创建MySQL数据库和创建表的详细步骤(navicat)_云边的快乐猫的博客-CSDN博客

三、项目配置

(一)pom.xl导入相关依赖 

这里面有几个必须的依赖

1.web依赖,这个是创建时候就已经加入进去了,所以不用导入了(检查一下)

2.lombok依赖 ,这个是创建时候就已经加入进去了,所以不用导入了(检查一下)

3.mybatis-plus的依赖(来源baomidou)

4.mysql依赖

5.druid依赖(来源阿里巴巴德鲁伊)

6.swagger2依赖(拥有这个,注解@ApiModelProperty才能用)

1.导入依赖

在pom.xml文件里面导入相关的依赖。如果依赖加入不成功,就要检查自己的maven配置问题了

maven解决方案:

springboot添加maven环境详细步骤_云边的快乐猫的博客-CSDN博客

spring-boot-maven-plugin报红的解决办法_云边的快乐猫的博客-CSDN博客

       <!--新加入的依赖1-->
		<dependency>
			<groupId>com.baomidou</groupId>
			<artifactId>mybatis-plus-boot-starter</artifactId>
			<version>3.4.3</version>
		</dependency>
		<!--新加入的依赖2-->
		<dependency>
			<groupId>mysql</groupId>
			<artifactId>mysql-connector-java</artifactId>
			<version>8.0.33</version>
		</dependency>
		<!--新加入的依赖3-->
		<dependency>
			<groupId>com.alibaba</groupId>
			<artifactId>druid-spring-boot-starter</artifactId>
			<version>1.2.16</version>
		</dependency>
		<!--新加入的依赖4  @ApiModelProperty这个才能用-->
		<dependency>
			<groupId>io.springfox</groupId>
			<artifactId>springfox-swagger2</artifactId>
			<version>2.7.0</version>
		</dependency>

 即:SpringBoot+Mybatis-Plus实现增删改查超详细步骤,spring boot,mybatis,后端,mybatis-plus,增删改查,结构讲解

(二)yml文件中配置连接数据库

2.配置yml文件 

把application配置文件的后缀改为yml,这样看着更分明一点,然后再里面写入连接数据库的配置,建立和数据库的关联。具体的端口号默认是8080的,但是我改成了80端口了。 

这里的第2、7、8、9行都要改为自己的  

SpringBoot+Mybatis-Plus实现增删改查超详细步骤,spring boot,mybatis,后端,mybatis-plus,增删改查,结构讲解

server:
  port: 80
spring:
  datasource:
    type: com.alibaba.druid.pool.DruidDataSource
    driver-class-name: com.mysql.cj.jdbc.Driver
    url: jdbc:mysql://localhost:3306/dndata?serverTimezone=GMT%2b8
    username: root
    password: 123456

四、代码的编写

数据库展示

SpringBoot+Mybatis-Plus实现增删改查超详细步骤,spring boot,mybatis,后端,mybatis-plus,增删改查,结构讲解

项目提前展示!!!

SpringBoot+Mybatis-Plus实现增删改查超详细步骤,spring boot,mybatis,后端,mybatis-plus,增删改查,结构讲解

(三)MySQL表绑定到spring boot(实体层)

为什么要建立这个类?

可以这样理解,这一步骤是让数据库字段和后端代码的绑定 

3.1 创建实体

在项目下新建一个entity包,再新建一个类。类的命名就是User

命名来源:类名和数据库中的表名一样,不过首字母要大写,一个属性类对应一张数据库表。

 3.2编写User类

这个实体类里面的属性要和数据库中的字段对应  

看着注释去编写,还是很简单的,在最下面的注释也解释了每个注解的作用 

User

package com.example.mybatis.entity;

import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableName;
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;

/*
* 1.使用@Data注解
* 2.使用注解@TableName(value = "user") -----user这个是对应数据库的表名
* 3.编写和数据库字段对应的属性
* 4.加上文档注解@ApiModelProperty("用户的id")---里面的文字是相当于给自己看的备注
* 5.使用@TableId(value = "id", type = IdType.AUTO) //作用是标明当前属性主键,并且type = IdType.AUTO这个代表id自增
* */
@Data  //1
@TableName(value = "user")    //2
public class User {

    @ApiModelProperty("用户的id")  //4
    @TableId(value = "id", type = IdType.AUTO) //5
    private Integer id;  //3

    @ApiModelProperty("用户名")  //4
    private String username;  //3

    @ApiModelProperty("密码")  //4
    private String password;   //3
}

/*
* 注解的解释
* @Data:注解是这个来源于lombok,内置了set、get、ToString等属性类里面我们需要写的东西,就不用我们写了

* @TableName:注解是 MyBatis-Plus 框架中的一个注解,用于标识实体类与数据库表之间的映射关系。
* 它的作用是告诉 MyBatis-Plus 框架这个实体类对应哪个数据库表。

* @ApiModelProperty:注解是 Swagger 框架中的一个注解,
* 用于给实体类的属性(字段)添加额外的文档说明,以便在生成 API 文档时提供更详细的描述和信息。
* Swagger是一个用于生成和展示 API 文档的工具,可以帮助开发人员更好地理解和使用 API。
*
* @TableId 注解用于标识表的主键字段,帮助 MyBatis-Plus 框架了解哪个属性在数据库中扮演主键的角色,
* 以及如何生成主键值。这对于数据库操作和映射非常重要。
* */

(四)springboot绑定到mybatis-plus(数据层)

为什么要创建这个接口呢?

因为为了把springboot获得的属性类再绑定给到mybatis-plus那边,然后绑定的这个接口给其他层调用。

4.1创建mapper包

项目下创建一个mapper包

4.2编写UserMapper接口

创建一个接口UserMapper,命名来源是属性类名+mapper,驼峰命名

到这里就可以使用mybatis-plus内置最原始的增删改查,但是不够灵活,只能做最简单的增删改查。

UserMapper

package com.example.mybatis.mapper;

import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.example.mybatis.entity.User;
import org.apache.ibatis.annotations.Mapper;
/*
* 1.使用@Mapper注解 ,代表这个接口被mybatis接管
* 2.继承BaseMapper<属性类名>
* */
@Mapper
public interface UserMapper extends BaseMapper<User> {
}


/*
* @Mapper:注解是 MyBatis 框架中的一个重要标识,
* 它定义了 Mapper 接口,用于与数据库交互。使用这个注解可以简化数据库操作代码,并提供一些优势,
* 如自动生成 SQL、类型安全性等。
* */

(五)把数据层加工处理成逻辑业务(service服务层)

为什么要这一层?

答:这一层次叫服务层,也叫业务层。因为上面的数据完全都绑定好了,可以实现增删改查了,但是那只是最原始的,封装的方法都很少,稍微复杂一点的分页查询方法里面都没有,所以要在这一层使用别人封装好的更好的方法以及各种逻辑都可以在这里写 

5.1创建service包 

SpringBoot+Mybatis-Plus实现增删改查超详细步骤,spring boot,mybatis,后端,mybatis-plus,增删改查,结构讲解

5.2编写获取内置增删改查方法接口IUserService

MP官方的

为什么要写这个接口?

继承了MP(mybatis-plus)封装好的接口,我们需要的各种增删改查方法都在这里面给我们写好了,这个接口编写是为了给后面控制层去调用这个接口里面的增删改查方法。

你说的增删改查方法怎么看不到?

因为被MP封装到了 这个IService接口里面了

在service包下建立这个IUserService接口

IUserService接口

package com.example.mybatis.service;

import com.baomidou.mybatisplus.extension.service.IService;
import com.example.mybatis.entity.User;


//1.继承IService这个接口,<实体类名>
public interface IUserService extends IService<User> {
}

/*
IService接口里面有mybatis-plus封装好我们经常会用到的增删改查的一些方法
里面本质上还调用了上数据层mapper,是对mapper的封装优化
*/
5.3编写自己定义的UserService

 在service包下建立这个UserService类

MP官方+自定义逻辑+自己调用数据层---咱自己的

5.2不是已经有可以调用增删改查的方法了吗,为什么还要写这个?

答:5.2那个接口是直接用MP内置的增删改查方法,可以正常用,但是我想对内置的这些增删改查再优化呢,比如数据的加密、数据的校验等逻辑处理。

而且我想根据我自己的需求来去给这些内置的增删改查嵌套或者升级自定义呢?

总而言之,这个是类底层是就是5.2接口的实现类,也就是说这个类包含了5.2的接口。还能自定义方法和对mapper数据层的调用,就比5.2接口的方法更加丰富,用于加上自定义处理繁琐的逻辑数据层

package com.example.mybatis.service;

import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.example.mybatis.entity.User;
import com.example.mybatis.mapper.UserMapper;
import org.springframework.stereotype.Service;

/*
* 1.使用@Service注解,要不然咱自己定义的类,谁知道他是服务层
* 2.继承ServiceImpl,这个是MP内置增删改查接口的一个实现,牛逼的是还能自定义直接调取数据层的数据
* 3.定义需要的方法,根据自己的需要定义,定义MP内置里面没有的
* */
@Service
public class UserService extends ServiceImpl<UserMapper, User> {
    //下面这个方法是对数据库操作时候多加了一层封装,用不到去掉也没有影响。
    // 、作用可以定义密码加密、数据校验、关联数据处理、日志记录、通知或触发事件等
    public boolean saveUser(User user){
           return saveOrUpdate(user);
    }
}
5.4编写服务层合成逻辑业务UserServiceImpl

 在service包下建立这个UserServiceImpl类 

1.为什么要写这个实现类?

因为把5.2和5.3的方法都拿来这里组成逻辑业务,比如登录、校验、权限等

2.为什么不直接用5.3,那个里面不是也可以定义逻辑方法吗? 

5.3那个自定义类主要用来处理一些对数据比较底层的方法,密码加密,校验这些底层。而这个类里面定义的是靠5.3那些小方法组起来的业务,比如登录、权限验证这些

总而言之,这里是业务的成型,调用5.3(主要)+5.2(次要)+一些逻辑组成业务

package com.example.mybatis.service.Impl;

import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.example.mybatis.entity.User;
import com.example.mybatis.mapper.UserMapper;
import com.example.mybatis.service.IUserService;
import org.springframework.stereotype.Service;
/*
* 1.@Service--------使用@Service注解,让人知道这个是服务层
* 2.extends ServiceImpl<UserMapper, User> ---继承和UserService类里面继承的方法一样,
* 但是这里面还偷偷藏了UserService自定义的方法
* 3.implements IUserService---实现接口
* */

@Service
public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements IUserService {
    //登录业务
    //登录校验业务
    //创建用户检查业务等等
}
5.5故事解读服务层接口和类

这么多服务层到底什么关系啊,这就来解读 

SpringBoot+Mybatis-Plus实现增删改查超详细步骤,spring boot,mybatis,后端,mybatis-plus,增删改查,结构讲解

这三个都是服务层的,看需要选择调用哪个

5.2(地主家傻儿子)是一个服务接口,去嵌套了MP内置的常用CRUD

     地主家MP的傻儿子一样,什么都没有,只会调用他老爹写好的方法,不过有个好老爹,大部分常用的CRUD他都有了,控制层可以直接调用他

5.3(佣人的儿子)是一个服务类,嵌套内置CRUD+自研调用方法

     这个类虽然和地主家傻儿子拥有一样的内置方法,但是却更上进,想要变得更强,所以可以自定义修炼很多方法:底层密码加密、数据校验、关联数据处理、日志记录、通知或触发事件,还能直接和数据层进行交互。

这个类厉害吧?是不是很完美?控制层不会调用他的,因为他这些方法都会被他爹实现类去组成各种逻辑业务

5.4(佣人)是一个实现服务类,继承有5.2地主家傻儿子,也实现了5.3亲儿子。大一统

      在这里面实现各种业务。能在这里编写的都是业务了,比如拿5.3方法组成的登录业务、权限校验业务等等。

这个是老爹服务类。够厉害了吧。但是这个也不能被控制层直接调用,因为这个不是接口,根据设计的依赖倒置原则,调用接口可以达到解耦的效果。但是这些写的方法又不能浪费,只能继承了5.2地主家的傻儿子接口,让他帮我把这些逻辑都带出去。通过控制层构造器注入

(六)调用服务层的皇帝

为什么要这一层?

因为叫控制层,是皇帝。你想要什么,告诉我,我让服务层去弄逻辑业务。

要什么数据或者逻辑业务都要通过我控制层,直接找我拿就好了。

既然服务层都有了,那我为什么要你?

服务层的业务只有逻辑业务,涉及到导入导出这种正常的业务,还得是我来

那该去调用服务层的哪个?

找这个接口-----IUserService

这里面不但继承了地主MP的CRUD,还继承了实现类里面所有的业务逻辑

 6.1创建controller包

SpringBoot+Mybatis-Plus实现增删改查超详细步骤,spring boot,mybatis,后端,mybatis-plus,增删改查,结构讲解

6.2 编写调取业务层的UserController类

 第一种理解方式

增删改查方法通用公式:

    1

public  2  自定义类名(    4的数据类型      )

   return   当前的全局服务类.服务类中的方法(4

服务中的方法:查全部:list根据ID查询:getById新增和修改saveOrUpdate删除:removeById

1.数据查询一般用@GetMapping、对数据进行修改操作用@PostMapping、删除用@PostMapping。

前端:我要删除或者根据id查询,

后端:那你get给我id就好了,其他的交给我,就是在注解的后面加("/{id}")。如果是直接查询全部的,你连id都不用给我,我懂

前端:我要修改和增加东西,有很多字段的数据都要

后端:那你不能放请求头里面,不安全,用post请求,放到请求体里面吧

2.这个是希望后端给我的数据,增删改什么的只要告诉我是否成功就好,所以用布尔值。查询你就要告诉我查询到什么了,就用属性类名

3.这个是和1相关,post请求用@RequestBody,其他的都用@PathVariable去解析请求头的id变成json传给后端

4.这个也看1,前端传什么数据过来,这里就用什么数据。什么都没有写就是全部---就是属性类名

第二种理解方式

@注解 ---这个就是前端要给后端的传输定义

public  希望后端给前端的   自定义类名  ( post注解使用@RequestBody-反之   数据类型    对应前端传过来的  )

         return  当前的全局服务类.服务类中的方法(数据类型对应前端传过来的)

package com.example.mybatis.controller;

import com.example.mybatis.entity.User;
import com.example.mybatis.service.IUserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;

import java.util.List;
/*
* 1.@RestController---添加控制层注解,让springboot知道这是控制层
* 2.@RequestMapping("/user")--- user代表默认访问路径在/user下可以访问
* */

@RestController
@RequestMapping("/user")
public class UserController {


    @Autowired  //4.自动注解
    private IUserService userService;  //3.定义调用服务层的接口


    /*
    *5.查询全部的数据
       5.1使用 @GetMapping注解,没有括号不定义就代表使用get方式,url访问上面默认路径/user
       5.2查询全部肯定使用的方法是集合<实体类>方式去自定义一个findAll方法名
       5.3再用服务层返回一个集合就好了
     */
    @GetMapping
    public List<User> findAll(){
        return userService.list();
    }


    /*6.根据id查询一条
      6.1查询使用@GetMapping注解,既然是根据id查询,那么就要把id传给后端。但是get是不能传递请求体的,
       只能传递请求头,那么只能把这个值放在请求头上面,就是这样("/{id}")放
      6.2自定义一个方法名(@PathVariable 类型 id),为什么要用id呢?这不是你前端这个"/{id}"说要吗
      @PathVariable这个注解的方式就是把请求头中的数字以Json传递给后端去识别,后端不能直接识别url上面的数字
      6.3 再用服务层返回获取id的方法就好了
    */
    @GetMapping("/{id}")
    public User findById(@PathVariable Integer id){
        return userService.getById(id);
    }

    /*7.新增和修改
         需要传递body传递全部参数,
        7.1对数据的操作使用@PostMapping,括号里面不定义就代表使用post方式,url访问上面默认路径/user
        7.2自定义一个方法,括号里面使用(@RequestBody 属性类 自定义属性昵称)
        为什么这里用Boolean而不是User,因为对数据的修改希望返回的结果就两种,要么成功,要么失败,所以用布尔值
        @RequestBody这个注解用于映射传递来的json参数映射为到java对象里面,特别是适用与请求体的参数
        7.3然后再用服务层返回获取的saveOrUpdate方法就好了
    */
    @PostMapping
    public Boolean add(@RequestBody User user){
        return userService.saveOrUpdate(user);
    }
    /*
    8.删除的方法
     8.1使用@DeleteMapping注解,因为删除只需要传递一个特定的id给后端就可以了,
     所以括号里面使用("/{id}")
     8.2创建一个自定义的方法,删除也是要么成功要么失败,所以使用Boolean作为返回值
     @PathVariable在上面已经解释过了
     8.3再用服务层次返回对应的方法,括号里面的值就是上面括号里面需要的值
    * */
    @DeleteMapping("/{id}")
    public Boolean delete(@PathVariable Integer id){
        return userService.removeById(id);
    }

}








/*    //修改数据的另一种写法
    @PostMapping("/update")
    public Boolean update(@RequestBody User user){
        return userService.updateById(user);
    }*/

到这里代码就全部编写完成了,不懂的去看上面的代码提前展示图和注释解释

MySQL--->属性类--->数据层mapper接口--->业务服务层service接口--->控制层(前后端交互) 

按照规范,不能跨级调用接口,属性类除外

五、前端请求测试

测试的工具可以使用idea自带的,也可以使用postman等请求测试工具。

postman官网下载安装登录详细教程_云边的快乐猫的博客-CSDN博客

具体的测试方法

postman测试后端增删改查_云边的快乐猫的博客-CSDN博客文章来源地址https://www.toymoban.com/news/detail-644384.html

到了这里,关于SpringBoot+Mybatis-Plus实现增删改查超详细步骤的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • springboot+mybatis-plus实现自动建表

    好长时间没输出了,最近工作上也是太多事,领导动不动就拍脑门,那叫一个酸爽~ 工作能力的提现不但是技术或解决问题的能力上,还体现在要能立刻满足领导的各种需求,不管是哪方面的需求,这样才能够拍上马屁,步步高升。 言归正传,作为技术从业者,还是要多深耕

    2024年02月16日
    浏览(37)
  • springboot~mybatis-plus的DynamicTableNameInnerInterceptor实现分表

    DynamicTableNameInnerInterceptor是mybatis-plug的一个拦截器插件,可以自己定义需要拦截的表单,然后对它进行加工,这时mybatis-plus就会把SQL代码的表名加上你的这个装饰。 我们通常把mybatis做成一个包,公司其它同事直接使用咱们的包,包里会统一定义数据 基类 、 数据分页 、 数据

    2024年02月06日
    浏览(53)
  • springboot mybatis-plus swing实现报警监听

    通过声音控制报警器,实现声光报警,使用beautyeye_lnf.jar美化界面如下 其他工具类

    2024年01月19日
    浏览(46)
  • SpringBoot 结合 mybatis-plus 实现分页功能

            要实现分页功能方法有很多,但最基本的实现原理都差不多,所以在实现功能之前要先把原理搞明白。正如删除有 “逻辑删除”  和 “物理删除\\\" 之分,分页也有 “逻辑分页” 和 “物理分页”; 1、逻辑分页:逻辑分页依赖于代码。(Mybatis自带的分页插件就是逻

    2024年02月06日
    浏览(42)
  • 化繁为简,MyBatis-Plus 里面的增删改查

    在当前盛行的 SpringBoot 项目中,整合持久层这一块,目前主流的有两种:JPA 和 MyBatis-Plus。至于哪个用的更多一些,这个主要还是看每个公司的技术架构,但硬是要说一个最为常用的,我认为是 MyBatis-Plus,而在这里也是对 MyBatis-Plus 的一个使用进行演示 好了,废话不多说,直

    2024年02月04日
    浏览(34)
  • Mybatis-plus中的DML编程控制--增删改的高级操作

    1.1、id生成策略控制 不同的表应用不同的id生成策略 日志:自增(1,2,3,4,…) 购物订单:特殊规则( FQ23948AK3843) 外卖单:关联地区日期等信息(10 04 20200314 34 91) 关系表:可省略id ······ 名称: @TableId 类型: 属性注解 位置:模型类中用于表示主键的属性定义上方 作用:设置当前类中主

    2024年02月16日
    浏览(43)
  • SpringBoot整合Mybatis-Plus、Jwt实现登录token设置

    Spring Boot整合Mybatis-plus实现登录常常需要使用JWT来生成用户的token并设置用户权限的拦截器。本文将为您介绍JWT的核心讲解、示例代码和使用规范,以及如何实现token的生成和拦截器的使用。 一、JWT的核心讲解 JWT(JSON Web Token)是一种基于JSON的,用于在网络上安全传输信息的

    2024年02月02日
    浏览(66)
  • SpringBoot整合mybatis-plus实现分页查询(建议收藏)

    一、前言         最近学习了SpringBoot分页查询的两种写法,一种是手动实现,另一种是使用框架实现。现在我将具体的实现流程分享一下。 二、手动实现分页查询         先复习一下,SQL中的limit,下面一行sql语句的意思是从第二个数据开始查,查询出两条数据

    2024年01月16日
    浏览(68)
  • SpringBoot整合Mybatis-Plus+Druid实现多数据源

    🌺本文主要讲解 springboot +mybatisplus + druid 实现多数据源配置功能 🌺 主页传送门:📀 传送 Spring Boot:    Spring Boot是一个基于Spring框架的开源Java开发框架,旨在简化Spring应用程序的开发、配置和部署。它提供了一种快速、敏捷的方式来构建独立的、生产级别的Spring应用程

    2024年02月09日
    浏览(102)
  • SpringBoot+Sharding-jdbc+mybatis-plus实现水平分表

    这块我就不演示了

    2024年02月12日
    浏览(32)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包