SpringBoot 整合mybatis-plus 高级功能及用法详解

springboot 整合mybatis-plus 高级功能及用法详解

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

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

image-20200806214645476

引入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.Driverurl: jdbc:mysql://localhost:3306/mybatis-plus?serverTimezone=UTC&useUnicode=true&characterEncoding=utf-8username: rootpassword: root# mybatis-plus相关配置
mybatis-plus:# 以下配置均有默认值,可以不设置global-config:db-config:#主键类型 auto:"数据库ID自增"id-type: autoconfiguration:# 是否开启自动驼峰命名规则映射:从数据库列名到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 文件夹

image-20200806215534085

(一)AR模式使用

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

即直接使用实体类 CRUD操作

1.继承Model

image-20200806221012349

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

image-20200806220842130

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

2.mapper接口

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

image-20200806221302376

那么,咱们来编写mapper

image-20200806221326120

再次测试

3.CRUD操作

新增操作

image-20200806221413677

查询操作

image-20200806221705933

修改操作

image-20200806221825358

删除操作

image-20200806221936204

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

(二)乐观锁

使用Mybatis-plus实现乐观锁

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

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

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

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

1.bean配置

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

image-20200806222513767

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

@Version
private Integer version;

image-20200806222639173

image-20200806222908272

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

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

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

首先插入一条数据

image-20200806223134584

image-20200806223150227

查询并修改该条信息

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

image-20200806224245623

image-20200806224258714

需要注意的几个点:

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

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

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

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

image-20200806224911876

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

image-20200806225056838

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

image-20200806225332135

如此,就实现了乐观锁了,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.数据库添加字段

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

image-20200808110331856

2.mp逻辑删除配置

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

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

image-20200808110905817

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

image-20200808111149966

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

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

image-20200808111441145

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

image-20200808111531657

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

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

image-20200808111933875

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

(四)自动填充

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

如下:

image-20200808112656976

实体类添加维护字段信息

image-20200808112809346

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

例如添加时

image-20200808113226978

image-20200808113243561

例如修改时

image-20200808113539478

image-20200808113559081

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

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

解决办法:

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

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

操作开始:

1.自定义实现类 MyMetaObjectHandler

这里需要注意哈

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

image-20200808114936751

@Slf4j
@Component
public class MyMetaObjectHandler implements MetaObjectHandler {@Overridepublic 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);}@Overridepublic 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)

image-20200808114731349

image-20200808114705528

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

image-20200808115231603

image-20200808115251910修改

image-20200808115517443

image-20200808115528774

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

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

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

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

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

3.自动填充优化

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

@Slf4j
@Component
public class MyMetaObjectHandler implements MetaObjectHandler {@Overridepublic void insertFill(MetaObject metaObject) {//判断是否有set属性if (metaObject.hasSetter("createId")) {//判断是否有值 如果没设置值 则为nullObject 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());}}}@Overridepublic 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需要维护,但我们设置了值 则不会进行填充了,修改亦是如此!

image-20200808124200342

image-20200808124331978

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

(五)数据保护

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

其和 Jasypt 项目加密方式类似

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

image-20200808154301136

image-20200808154406180

2.配置文件替换

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

格式:

mpw:xxxx密文信息

image-20200808154539290

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

idea 设置方式:

image-20200808154642114

image-20200808154702402

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

image-20200808154741030

jar 启动时 设置密钥参数:

image-20200808155350475

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/555111.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

8除以2表示什么意思_八字中劫财,比肩分别表示什么意思

◆◆比劫与其他十神的生克关系◆◆食伤&#xff1a;比劫生食伤财星&#xff1a;比劫克财星官星&#xff1a;官星克比劫印星&#xff1a;印星生比劫◆◆比劫帮身◆◆八字术语&#xff0c;比肩劫财均可助日干之力&#xff0c;如甲见甲(比)、乙(劫)、寅(禄)、卯(刃)之类&#xff0…

江苏省计算机一级考试基础知识,江苏省计算机一级考试大纲及复习指南

江苏省计算机一级考试大纲及复习指南江苏省计算机等级考试大纲(一级)计算机信息技术考试要求(试用)及使用说明1&#xff0e;掌握计算机信息处理与应用的基础知识。2&#xff0e;能比较熟练地使用六个常用的软件(IE、Outlook Express,Word,Excel,PowerPoint,和FrontPage)3掌握Ac…

MyBatis-plus 自动填充MetaObjectHandler不生效问题

背景 在项目中经常遇到一些数据需要自动填充&#xff0c;填充方法都是同一种规则&#xff0c;例如创建时间等。 1 自动填充方法配置 1.1 实体类添加注解 1.2 在handler包下实现元对象处理器接口 import com.baomidou.mybatisplus.core.handlers.MetaObjectHandler; import o…

html拖拽模态框,可拖拽的谷歌样式纯javascript模态窗口插件

draggable-google-modal是一款可拖拽的谷歌样式纯javascript模态窗口插件。该模态窗口插件使用纯js编写&#xff0c;模态窗口可以任意进行拖动&#xff0c;并且模态窗口可以全屏放大&#xff0c;或左右对齐占据半边屏幕。使用方法使用该模态窗口插件需要引入draggabilly.pkgd.j…

centos7修改网卡顺序_CentOS7网卡名称的修改

生产环境中&#xff0c;托管于机房的服务器经常会进行割接操作&#xff0c;为了能让网卡更容易让运维人员和机房识别&#xff0c;避免因为弄错网线或找错网卡导致业务中断&#xff0c;在服务器上架前就应该对所有网卡按照 eth0、eth1 的方式来命名&#xff0c;并按照网卡的物理…

new Gson().toJson日期转特定格式日期实体

Google的Gson功能非常强大&#xff01; 格式化日期我们只需要这样创建就好了 Student stu new Student();stu.setName("张三");stu.setBirthDay(new Date());如果张三的生日是日期型的话&#xff0c;直接实体转json的话&#xff0c; new Gson().toJson(stu);日期…

计算机考试internet应用好考吗,2015年职称计算机考试XP好考还是internet应用好考...

WindowsXP和Internet的相似考点Windows XP模块和Internet模块是两个常考模块&#xff0c;2010年7月以来&#xff0c;国家职考题库发生了较大的变动&#xff0c;之前我们已经从题目上总结出了两个的模块的类似点&#xff0c;现在从大纲上总结两个模块的相似考点。一&#xff1a;…

博途变量类型_PLC数据类型(UDT)

UDT类型是一种由多个不同数据类型元素组成的数据结构&#xff0c;元素可以是基本数据类型&#xff0c;也可以是STRUCT、数组等复杂数据类型以及其它UDT等。UDT类型嵌套UDT类型的深度限制为 8 级。UDT类型可以在DB、OB/FC/FB接口区处使用。从TIA 博途 V13SP1 开始&#xff0c;S7…

Mybatis-Plus条件参数查询手册

【引言】 使用mybatis-plus框架的基础上&#xff0c;直接使用其中的条件参数进行查询还是很方便的。但每次使用到像大于、不等于这样一些不常用条件时&#xff0c;都需要现查&#xff0c;所以记录在这篇博客里&#xff0c;当作一个自己的查询手册。 【手册】 查询方式说明se…

只考计算机知识吗,计算机二级只考一门吗?计算机二级考什么?

计算机二级只考一门吗&#xff1f;计算机二级只需要考一门。合格就可以获得国家二级计算机证。基础知识和操作题是一起考的&#xff0c;国家二级计算机考试是无纸化考试。另外&#xff0c;基础知识都是选择题&#xff0c;操作题包括主要的办公软件的操作以及高级应用。相对于电…

java 最快平衡几个值_Java 集合框架面试问题集锦

Java集合框架(例如基本的数据结构)里包含了最常见的Java常见面试问题。很好地理解集合框架&#xff0c;可以帮助你理解和利用Java的一些高级特性。下面是面试Java核心技术的一些很实用的问题。Q&#xff1a;最常见的数据结构有哪些&#xff0c;在哪些场景下应用它们&#xff1f…

mybatis-plus 如何判断参数是否为空并作为查询条件

判断参数是否为空并作为查询条件 Overridepublic Page<DemandEntity> selectByDepartmentDisplay(DemandEntity demandEntity) {EntityWrapper<DemandEntity> wrapper new EntityWrapper<DemandEntity>();wrapper.eq(!StringUtils.isNullOrEmpty(demandEnt…

html超链接使用d,HTML图像的调用和超链接

1.HTML中图像的使用语法&#xff1a;属性 &#xff1a;src &#xff1a;指定要显示图像的URLwidth &#xff1a;图像的宽度 (默认等比例变化)height &#xff1a;图像的高度title &#xff1a;鼠标悬停时&#xff0c;要显示的文本alt &#xff1a;图片加载失败后现实的文字2.超…

linux 总数 进程_Linux运用一些常用命令,优秀的PHPer都需掌握

作为一名优秀的phper&#xff0c;Linux是必备的一项技能&#xff0c;工作3-5年的基本能明白我讲的道理&#xff01;今天搜集整理了一些Linux服务器运维常用命令,希望对大家有帮助&#xff1a;1.删除0字节文件find -type f -size 0 -exec rm -rf {} 2.查看进程按内存从大到小排列…

Map中的computeIfAbsent方法

文章目录1、概述2、代码示例2.1 示例12.2 示例22.2 示例31、概述 在JAVA8的Map接口中&#xff0c;增加了一个computeIfAbsent&#xff0c;方法签名如下&#xff1a; default V computeIfAbsent(K key, Function<? super K, ? extends V> mappingFunction)Function是一…

计算机系统结构结构相关实验报告,计算机系统结构实验报告.doc

电 子 科 技 大 学实 验 报 告学生姓名&#xff1a; 学 号&#xff1a; 一、实验室名称&#xff1a;计算机学院实验中心二、实验项目名称&#xff1a;计算机系统结构课程实验三、实验原理&#xff1a;1. 由教师给出一个具有基本功能的Verilog HDL流水线源代码&#xff1b;2. 学…

c语言高低位拷贝_C语言指针详解

1为什么使用指针 假如我们定义了 char a’A’ &#xff0c;当需要使用 ‘A’ 时&#xff0c;除了直接调用变量 a &#xff0c;还可以定义 char *p&a &#xff0c;调用 a 的地址&#xff0c;即指向 a 的指针 p &#xff0c;变量 a&#xff08; char 类型&#xff09;只占了一…

音乐计算机怎么用,如何利用电脑录制自己的歌曲(音乐)

一、 如何利用电脑录制自己的歌曲把自己在录音过程中的一些心得和体会告诉大家&#xff0c;希望大家能通过此贴&#xff0c;很快的掌握cooledit pro&#xff0c;并录出自己满意的歌曲。也许你看过之后&#xff0c;会觉得打造一个专业级的声音竟是如此的简单。首先&#xff0c;我…

MybatisPlus中@TableField注解的使用详解

MybatisPlus中TableField注解的使用详解 实现 官方文档说明&#xff1a; com.baomidou.mybatisplus.annotations.TableField TableField注解新增属性 update 预处理 set 字段自定义注入 (讲解&#xff1a;比如我们使用mybatisplus自带的insert()方法向数据库插入数据时&…

dataframe记录数_大数据系列之Spark SQL、DataFrame和RDD数据统计与可视化

Spark大数据分析中涉及到RDD、Data Frame和SparkSQL的操作&#xff0c;本文简要介绍三种方式在数据统计中的算子使用。1、在IPython Notebook运行Python Spark程序IPython Notebook具备交互式界面&#xff0c;可以在Web界面输入Python命令后立刻看到结果&#xff0c;还可将数据…