springboot2.3 整合mybatis-plus 高级功能(图文详解)

学习并使用mybatis-plus的一些高级功能的用法例如: AR模式、 乐观锁 、逻辑删除 、自动填充、数据保护等功能

为了方便演示,咱们还是新建一个全新的项目

引入mp依赖

<dependency>
 <groupId>com.baomidou</groupId>
 <artifactId>mybatis-plus-boot-starter</artifactId>
 <version>3.3.2</version>
 </dependency>

yml配置

# 配置端口
server:
 port: 8080
spring:
 # 配置数据源
 datasource:
 driver-class-name: com.mysql.cj.jdbc.Driver
 url: jdbc:mysql://localhost:3306/mybatis-plus?serverTimezone=UTC&useUnicode=true&characterEncoding=utf-8
 username: root
 password: root

# mybatis-plus相关配置
mybatis-plus:
 # 以下配置均有默认值,可以不设置
 global-config:
 db-config:
 #主键类型 auto:"数据库ID自增"
 id-type: auto
 configuration:
 # 是否开启自动驼峰命名规则映射:从数据库列名到Java属性驼峰命名的类似映射
 map-underscore-to-camel-case: true
 # 如果查询结果中包含空值的列,则 MyBatis 在映射的时候,不会映射这个字段
 call-setters-on-nulls: true
 # 这个配置会将执行的sql打印出来,在开发或测试的时候可以用
 log-impl: org.apache.ibatis.logging.stdout.StdOutImpl

在 Spring Boot 启动类中添加 @MapperScan 注解,扫描 Mapper 文件夹

(一)AR模式使用

ActiveRecord模式:支持 ActiveRecord 形式调用,实体类只需继承 Model 类即可进行强大的 CRUD 操作

即直接使用实体类 CRUD操作

1.继承Model

点进Model 类中 ,发现其提供了一些基础的CRUD操作方法,并实现了序列化接口

注意的是,如果要使用ActiveRecord模式,仅仅继承Model 是不行的,这一点官网没有很显示的提到,还需要编写mapper接口 继承BaseMapper接口,泛型为当前实体类

2.mapper接口

如果不编写Mapper直接使用Model方法会报错:

那么,咱们来编写mapper

再次测试

3.CRUD操作

新增操作

查询操作

修改操作

删除操作

此AR模式呢,可以在开发中减少我们很多的调用代码,简单操作无需额外使用Mapper、service调用

(二)乐观锁

使用Mybatis-plus实现乐观锁

乐观锁:总是假设最好的情况,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据,乐观锁适用于多读的应用类型,这样可以提高吞吐量。

mybatis-plus 提供的乐观锁 是采用了版本号机制

数据表中加上一个数据版本号version字段,表示数据被修改的次数,当数据被修改时,version值会加一。当线程A要更新数据值时,在读取数据的同时也会读取version值,在提交更新时,若刚才读取到的version值为当前数据库中的version值相等时才更新,否则重试更新操作,直到更新成功。

mp项目使用乐观锁是使用其插件配置–配置Bean的形式

1.bean配置

@Bean
public OptimisticLockerInterceptor optimisticLockerInterceptor() {
 return new OptimisticLockerInterceptor();
}

2.实体类以及数据库添加对应字段以及列

@Version
private Integer version;

至于version初始值,可以数据库默认设置为0,也可以再新增数据时手动设置verion版本号,个人是采用了数据库默认值

那么我们在做任何查询修改删除的时候呢,mp会默认帮我们吧版本号作为条件带上,判断与数据库中该数据版本号是否一致。

3.相关操作以及注意事项

首先插入一条数据

查询并修改该条信息

发现其在修改的时候,将version作为了条件,并对version做了修改 set操作(默认是在原基础上+1)

需要注意的几个点:

支持的数据类型只有:int,Integer,long,Long,Date,Timestamp,LocalDateTime
整数类型下 newVersion = oldVersion + 1
newVersion 会回写到 entity 中
仅支持 updateById(id) 与 update(entity, wrapper) 方法
在 update(entity, wrapper) 方法下, wrapper 不能复用!!!

什么意思呢,就是要想使用乐观说,首先 类型有限制,其次,修改方法仅仅只有上方两个有效,才会在修改时候对版本号就行修改操作 以及 要想版本号升级必须 是要把之前版本号传过去。

即version操作时,我必须把旧的带过去

版本号不对,是无法进行数据更新操作的(删除、修改)

而如果不传递版本号的话,那么该字段就不会被维护了(失去了乐观锁 版本号机制 的意义了)

版本号回传 修改正确 版本号维护成功

如此,就实现了乐观锁了,mp基础上使用乐观锁,就是这么简单

(三)逻辑删除

物理删除:将数据库中该信息进行彻底删除,无法恢复。 应的SQL语句:delete from 表名 where 条件

逻辑删除:逻辑删除的本质是修改操作,所谓的逻辑删除其实并不是真正的删除,而是在表中将对应的是否删除标识,然后修改 查询操作时将 是否删除标识字段作为条件带上,进而达到逻辑上删除了数据,但实际仍然保留了数据(尽管此数据在逻辑中不会再使用了)

例如:自己设置一个逻辑删除字段,例如1表示数据有效,0表示被删除即可,默认是1

Mybatis-plus 已经为我们提供了逻辑删除封装

只对自动注入的sql起效:

插入: 不作限制
查找: 追加where条件过滤掉已删除数据,且使用 wrapper.entity 生成的where条件会忽略该字段
更新: 追加where条件防止更新到已删除数据,且使用 wrapper.entity 生成的where条件会忽略该字段
删除: 转变为 更新
例如:

删除: update user set deleted=1 where id = 1 and deleted=0
查找: select id,name,deleted from user where deleted=0
字段类型支持说明:

支持所有数据类型(推荐使用 Integer,Boolean,LocalDateTime)
如果数据库字段使用datetime,逻辑未删除值和已删除值支持配置为字符串null,另一个值支持配置为函数来获取值如now()
附录:

逻辑删除是为了方便数据恢复和保护数据本身价值等等的一种方案,但实际就是删除。
如果你需要频繁查出来看就不应使用逻辑删除,而是以一个状态去表示。

使用步骤:

1.数据库添加字段

对某个表的数据需要使用逻辑删除,则必须创建逻辑删除字段

2.mp逻辑删除配置

在原有的mp配置上 添加逻辑删除设置

其已删除 未删除值均有默认值 1 0 如果你预想设置值与之一致,仅仅编写 逻辑删除字段名即可

3.实体类加逻辑删除字段以及注解

mp中 逻辑删除相关配置就没了 ,很是简单------------咱们开始测试

咱们先删除数据库中原有数据

可以看到 ,删除语句实际上是执行了update(修改) 额外将 我们设置好的逻辑删除字段值作为条件(未删除 0),并修改 值为1(已删除)

接下来,我们来进行查询操作,看是否还能取出数据

total=0 ,没有数据, 那么我们便做到了业务意义上的数据删除(逻辑删除)

其只要是设置了逻辑删除 ,修改 删除 查询操作均会携带上未删除条件 ,保证操作的数据时 “存在” 的()

(四)自动填充

在开发中哈,我们一个表的字段除了业务字段外,实际上还会添加维护字段 ,即 创建人 创建时间 修改人 修改时间 ,让数据的操作 有迹可循,方便管理

如下:

实体类添加维护字段信息

那么每次在操作数据的时候需要我们手动插入到该model中 进行保存,再操作数据

例如添加时

例如修改时

这样呢,每次我们在操作数据时都需要在业务代码中编写 新增用户ID 当前时间 修改用户ID 修改时间等字段等,就比较繁琐

项目很大,几百个Model 乘以Four 那么就额外编写几千行代码进行维护了

解决办法:

1.自定义Aop 进行数据填充

2.使用现成的 (mp都用了,用他的自动填充不香吗)

操作开始:

1.自定义实现类 MyMetaObjectHandler

这里需要注意哈

这二者呢,一个是插入的类型 一个是插入的值 必须对应上,或者 自动填充时会报错

@Slf4j
@Component
public class MyMetaObjectHandler implements MetaObjectHandler {

 @Override
 public void insertFill(MetaObject metaObject) {
 log.info("start insert fill ....");
 this.strictInsertFill(metaObject, "createTime", Long.class, System.currentTimeMillis());
 //开发中 创建者 修改者ID 通过Aop 或者权限框架获取当前操作用户 ,用用户ID进行填充即可
 this.strictInsertFill(metaObject, "createId", Long.class, 666L);

 }

 @Override
 public void updateFill(MetaObject metaObject) {
 log.info("start update fill ....");
 this.strictUpdateFill(metaObject, "modifyTime", Long.class, System.currentTimeMillis());
 this.strictUpdateFill(metaObject, "modifyId", Long.class, 666L);

 }
}

2.实体类注解

要开启自动填充功能 实现 MetaObjectHandler 还不行,因为Mp 啊,对其字段默认是不做处理的,所以,咱们配置了 新增修改 自动填充时 还需要在其字段上进行开启

@TableField(fill = FieldFill.INSERT)

OK。设置好了,咱们插入修改试一试吧!

修改

如此 mp的自动填充就可以了,根据业务 合理对默写字段填充,填充什么值 自己考量即可

但是呢,也是有一定问题的

比如:我有一个定时维护功能,是由xxl-job发起的定时任务,那么这个时候,我字段维护信息,为了区分开到底是认为维护还是自动维护,那么就需要手动设置 创建修改人 以及时间了

这个时候呢,自动填充功能会把我们自己设置的进行覆盖

所以呢,自动填充咱们也可以进行优化,当有该维护字段时,且字段无值时才填充,当无维护字段或者 维护字段设置了值则不填充

3.自动填充优化

实际就是做了判断 判断该字段是否有set属性 有则尝试填充 (无值则填充)

@Slf4j
@Component
public class MyMetaObjectHandler implements MetaObjectHandler {

 @Override
 public void insertFill(MetaObject metaObject) {
 //判断是否有set属性
 if (metaObject.hasSetter("createId")) {
  //判断是否有值 如果没设置值 则为null
  Object createId = getFieldValByName("createId", metaObject);
  if (createId == null) {
  log.info("创建人需要维护 且值为空 需要填充 ");
  //开发中 创建者 修改者ID 通过Aop 或者权限框架获取当前操作用户 ,用用户ID进行填充即可
  this.strictInsertFill(metaObject, "createId", Long.class, 666L);

  }
 }
 if (metaObject.hasSetter("createTime")) {
  Object createTime = getFieldValByName("createTime", metaObject);
  if (createTime == null) {
  log.info("创建时间需要维护 且值为空 需要填充 ");
  this.strictInsertFill(metaObject, "createTime", Long.class, System.currentTimeMillis());
  }
 }

 }

 @Override
 public void updateFill(MetaObject metaObject) {
 if (metaObject.hasSetter("modifyId")) {
  Object modifyId = getFieldValByName("modifyId", metaObject);
  if (modifyId == null) {
  log.info("修改人需要维护 且值为空 需要填充 ");
  this.strictUpdateFill(metaObject, "modifyId", Long.class, 666L);
  }
 }
 if (metaObject.hasSetter("modifyTime")) {
  Object modifyTime = getFieldValByName("modifyTime", metaObject);
  if (modifyTime == null) {
  log.info("修改时间需要维护 且值为空 需要填充 ");
  this.strictUpdateFill(metaObject, "modifyTime", Long.class, System.currentTimeMillis());
  }
 }

 }
}

测试:

我这里设置了创建者ID ,虽然createId需要维护,但我们设置了值 则不会进行填充了,修改亦是如此!

那么自动填充功能,就完了,整合时 结合项目,适当修改即可。

(五)数据保护

使用mp中的加密方式,对配置文件中一些敏感信息进行加密(例如密码)

其和 Jasypt 项目加密方式类似

1.生成随机秘钥以及内容加密

2.配置文件替换

我们将原本的明文信息(root)换成加密后的密文信息

格式:

mpw:xxxx密文信息

3.设置密钥到启动参数中

idea 设置方式:

启动测试–获取到了数据 证明数据库连接成功,其加密内容在连接时也被密钥解密了,但配置中,无明文信息,一定程度保护了项目

jar 启动时 设置密钥参数:

项目地址:springboot-mp进阶

到此这篇关于springboot2.3 整合mybatis-plus 高级功能的文章就介绍到这了,更多相关springboot2.3 整合mybatis-plus内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • 解决Spring boot整合mybatis,xml资源文件放置及路径配置问题

    网上各种解决方案,我试了好久,整合了几篇文章才凑出来,在这里分享一下,实在不想网友们在这里面绕圈子,毕竟,写代码的时间是愉快的,解决bug也是愉快的,但也是一直在bug里面绕圈子就不爽了. 亲自试验: 1) 我的mapper和xml是这样子放置的 2) 在.xml中namespace是这样的: 3) application.properties中mybatis.mapper-locations得这么配置到xml 4) 最后呢,你只要在pom.xml中build下这样配置 5) 按照我这种位置防止

  • SpringBoot整合mybatis-generator-maven-plugin的方法

    前言: SpringBoot版本 : 2.2.6 mybatis-generator-maven-plugin版本: 1.4.0 plugin 使用 mysql版本: 5.1.47 本次主要记录日和使用该插件自动生成pojo mapper xml文件(单表) 1 创建SpringBoot工程 这里不做具体记录,创建工程往常教程很多, 下面贴出重要文件和注意事项 1.1 pom.xml <?xml version="1.0" encoding="UTF-8"?&

  • SpringBoot整合MybatisSQL过滤@Intercepts的实现

    场景: 系统模块查询数据库需要根据用户的id去筛选数据.那么如果在 每个sql加user_id的过滤显然不明确.所以要在查询前将sql拼接上条件,做统一管理. 开发环境: spring boot + mybatis 只需一个拦截类即可搞定(在看代码前需要了解注解@Intercepts()): @Component @Intercepts({ @Signature(type = Executor.class, method = "query", args = {MappedStateme

  • SpringBoot如何通过yml方式整合Mybatis

    本来打算写个使用Sharding-JDBC的例程,但是在搭建Mybatis的过程中,一波三折,因为好久没搭建项目了,另外加上换了电脑.所以很破折,在这里记录一下Spring Boot整合Mybatis吧.可能很简单,但是我长时间没用忘记了,我这里备忘一下吧. 一.项目目录结构 注意这里Application文件的位置,它是与controller.entity.mapper.service等包处于并列的关系. 二.数据库文件 SET NAMES utf8mb4; SET FOREIGN_KEY_C

  • springboot整合mybatis-plus 实现分页查询功能

    建一个config类 @Configuration public class MybatisPlusConfig { @Bean public PaginationInterceptor paginationInterceptor(){ return new PaginationInterceptor(); } } 编写controller post /article/search/{page}/{size} @PostMapping("search/{page}/{size}") p

  • 解决SpringBoot整合Mybatis扫描不到Mapper的问题

    闲来无事,想学学springboot,开始搭建一个项目,但是一直显示mapper扫描不到的错误: "Exception encountered during context initialization - cancelling refresh attempt: org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'userController': Unsa

  • Spring整合Mybatis的全过程

    1.Spring配置文件 1.1配置数据库连接池 <!--读取文件--> <util:properties id="config" location="classpath:Config/db.properties"/> <!--配置数据库连接池--> <bean id="source" class="org.apache.commons.dbcp.BasicDataSource">

  • Spring Boot整合mybatis使用注解实现动态Sql、参数传递等常用操作(实现方法)

    前面介绍了Spring Boot 整合mybatis 使用注解的方式实现数据库操作,介绍了如何自动生成注解版的mapper 和pojo类. 接下来介绍使用mybatis 常用注解以及如何传参数等数据库操作中的常用操作. 其实,mybatis 注解方式 和 XML配置方式两者的使用基本上相同,只有在构建 SQL 脚本有所区别,所以这里重点介绍两者之间的差异,以及增删改查,参数传递等注解的常用操作. 详解SpringBoot 快速整合Mybatis(去XML化+注解进阶)已经介绍过了,不清楚的朋友可

  • springboot2.3 整合mybatis-plus 高级功能(图文详解)

    -学习并使用mybatis-plus的一些高级功能的用法例如: AR模式. 乐观锁 .逻辑删除 .自动填充.数据保护等功能 为了方便演示,咱们还是新建一个全新的项目 引入mp依赖 <dependency> <groupId>com.baomidou</groupId> <artifactId>mybatis-plus-boot-starter</artifactId> <version>3.3.2</version> &l

  • SQL Server 2016 Alwayson新增功能图文详解

    概述 SQLServer2016发布版本到现在已有一年多的时间了,目前最新的稳定版本是SP1版本.接下来就开看看2016在Alwyson上做了哪些改进,记得之前我在写2014Alwayson的时候提到过几个需要改进的问题在2016上已经做了改进. 一.自动故障转移副本数量 在2016之前的版本自动故障转移副本最多只能配置2个副本,在2016上变成了3个. 说明:自动故障转移增加到三个副本影响并不是很大不是非常的重要,多增加一个故障转移副本也意味着你的作业也需要多维护一个副本.重要程度(一般).

  • 显示IntelliJ IDEA工具的Run Dashboard功能图文详解

    1.点击IDEA上面工具栏View-->Tool Windows-->Run Dashboard,如下图: 2.如果Tool Windows的子选项里没有找到Run Dashboard,可以在工程目录下找.idea文件夹下的workspace.xml,增加如下组件,如图: 代码: <component name="RunDashboard"> <option name="configurationTypes"> <set&g

  • IDEA的常见的设置和优化功能图文详解

    显示工具条 (1)效果图 (2)设置方法 标注1:View–>Toolbar标注2:View–>Tool Buttons设置鼠标悬浮提示 (1)效果图 (2)设置方法 File–>settings–>Editor–>General–>勾选Show quick documentation- 显示方法分隔符 (1)效果图 (2)设置方法 File–>settings–>Editor–>Appearance–>勾选 忽略大小写提示 (1)效果图 备注:

  • SpringBoot整合Mybatis与druid实现流程详解

    目录 SpringBoot整合junit SpringBoot整合junit SpringBoot整合junit的classes SpringBoot整合Mybatis 整合前的准备 整合Mybatis SpringBoot 整合druid 配置前置知识小点 整合druid SpringBoot整合junit SpringBoot整合junit ①还是一样,我们首先创建一个SpringBoot模块. 由于我们并不测试前端,而只是整合junit,所以不用选择模板,选择其中的web即可. 完成以后我

  • mybatis查询匹配机制图文详解

    第一种查询方式匹配机制 如何找到我们想要的sql语句 第一步: 先匹配sqlsqssion所在的MybatisUtil.java 第二步: 去匹配mybatis-config.xml 匹配mappers中的所有的xml文件的namespace, 如dao.RoleDao,再匹配sql语句的id 注意:如果这些xml文件之中有相同的namespace,接着去匹配id,如果id再相同,就会出现错误,系统不知道要匹配哪个 第二种查询方式匹配机制 利用反射去找RoleDao接口的class对象及其接口所

  • 微信公众平台开发教程③ PHP实现微信公众号支付功能图文详解

    本文实例讲述了PHP实现微信公众号支付功能.分享给大家供大家参考,具体如下: 直言无讳,我就是一个初涉微信开发的小白,写这篇博客的原因:一是为了给自己做下备忘记录,以便以后能回忆这条程序猿的坎坷路:二是希望能帮助到同是自学开发的小白们:三是对那些不屑一顾于我等尘埃的大牛们的控诉,小白的道路坎坷,你们凭什么总要一副高高在上的样子?我等敬而不畏... 背景介绍: 随着智能手机的普及,移动支付下的微信.支付宝所提供的便利需求不言而喻,好吧,至少我周围连个小摊贩的早餐都可以微信支付,而且人家手机还比我高

  • 微信公众平台开发教程④ ThinkPHP框架下微信支付功能图文详解

    本文实例讲述了ThinkPHP框架下微信支付功能.分享给大家供大家参考,具体如下: 声明:原文主要摘自白俊遥博客 ,部分内容针对个人事例已作修改,主要用于自己的参考,欢迎指正. 注意:微信公众号支付,强烈建议使用外网可访问的链接测试,否则即便代码正确也无法调出支付界面,可使用草料二维码生成可扫描图片,也可直接在微信中输入网址. 使用框架:ThinkPHP 3.2.3 一.微信公众平台信息配置 1).进入微信公众平台 由左侧的"微信支付"进入配置界面,添加或修改正确的支付授权目录,注意该

  • 微信小程序开发实现的选项卡(窗口顶部/底部TabBar)页面切换功能图文详解

    本文实例讲述了微信小程序开发实现的选项卡(窗口顶部/底部TabBar)页面切换功能.分享给大家供大家参考,具体如下: 微信小程序开发中选项卡.在android中选项卡一般用fragment,到了小程序这里瞬间懵逼了. 总算做出来了.分享出来看看. 先看效果: 再上代码: 1.index.wxml <!--index.wxml--> <view class="swiper-tab"> <view class="swiper-tab-list {{c

  • 微信公众平台开发教程②微信端分享功能图文详解

    本文实例讲述了微信公众平台微信端分享功能.分享给大家供大家参考,具体如下: 背景 初次尝试微信公众号的开发,对于学习方法的探索都是来源于网上的博客.问答,对于参差不齐的信息,自己也是有苦说不出,抽出一点时间写点文章,既是对自己的学习总结,也希望给予同是菜鸟的小白一点帮助. 今天想添加微信分享的功能,如果不进行自定义设计,那么当我们点击分享朋友圈.好友或者QQ好友.空间时,默认的标题就是<title>标签中的信息,而显示的描述信息就是链接,图片多是默认为页面中显示的第一张图片,显然这样的处理是不

随机推荐