百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

三分钟学会使用Mybatis-Plus——笔记

wptr33 2024-12-15 17:13 26 浏览

简介

MyBatis-plus是mybatis的增强工具,在MyBatis 上只做增强,不做改变,引入他不会对现有工程产生影响,只需简单配置快速实现CURD操作,从而节省大量时间。代码生成、自动分页、逻辑删除、自动、填充等功能一应俱全。

安装

首先创建一个springboot项目,导入Mybatis-plus相关依赖:

<dependency>

<groupId>mysql</groupId>

<artifactId>mysql-connector-java</artifactId>

<version>8.0.21</version>

</dependency>

<dependency>

<groupId>com.alibaba</groupId>

<artifactId>druid-spring-boot-starter</artifactId>

<version>1.1.10</version>

</dependency>

<dependency>

<groupId>com.baomidou</groupId>

<artifactId>mybatis-plus-boot-starter</artifactId>

<version>3.4.2</version>

</dependency>

由于Mybatis-plus是包含Mybatis 依赖的所以导入Mybatis-plus后就不用导入Mybatis了

配置

在springboot的yml文件中可以配置实体类别名和mapper文件路径:

mybatis-plus:

mapper-locations: classpath*:/mapper/**/*.xml

type-aliases-package: com.xxz.pojo

mapperlocations 自动配置好的,默认值是classpath*:/mapper/**/*.xml 意为任意包路径下所有的mapper包下的xml文件

也可以使用Mybatis中的方法进行配置,因为Mybatis-plus是只做增强不做改变。

实体类常用注解

@TableName("dept") 对应数据库的表明

@TableField("deptno") 对应数据库的字段名

@TableField(exist = false) 是否为数据库表字段

详细文档:注解 | MyBatis-Plus

@AllArgsConstructor

@NoArgsConstructor

@Data

@TableName("dept")//用于数据库表明和实体类名不一致情况

public class Dept implements Serializable {

/*@TableField(exist = false)

private String aaa;*/

@TableField("deptno")//用于数据库字段名和实体类属性不一致情况

private Integer deptno;

private String dname;

private String loc;

}

使用

mapper类需要实现BaseMapper<T>接口,传入实体类的泛型

public interface DeptMapper extends BaseMapper<Dept> {

}

service接口需要继承Iservice<T>类,传入实体类的泛型

public interface DeptService extends IService<Dept> {

}

service实现类需要继承ServiceImpl<M,T>类,传入mapper类,和实体类的泛型

@Service

public class DeptServiceImpl extends ServiceImpl<DeptMapper, Dept> implements DeptService {

}

CRUD接口

save方法:

// 插入一条记录(选择字段,策略插入)

boolean save(T entity); entity实体对象

// 插入(批量)

boolean saveBatch(Collection<T> entityList); entityList实体对象集合

// 插入(批量)

boolean saveBatch(Collection<T> entityList, int batchSize); batchSize插入批次数量

// 增加

@Test

public void testAdd(){

boolean save = deptService.save(new Dept(null, "aaa", "bbb"));

System.out.println(save);

}

list方法:

// 查询所有

List<T> list();

// 查询列表

List<T> list(Wrapper<T> queryWrapper);queryWrapper条件构造

// 查询集合

@Test

public void testQueryWrapper(){

// 部门号》=20

// QueryWrapper 作用就是在原本的SQL语句后面拼接where条件

// selec * from where delete from dept where update dept set ... where ....

QueryWrapper<Dept> queryWrapper=new QueryWrapper<>();

//queryWrapper.ge("deptno", 20).eq("dname", "ACCOUNTING").likeRight("dname", "A");

//queryWrapper.likeRight("dname", "A");

List<Dept> list = deptService.list(queryWrapper);

for (Dept dept : list) {

System.out.println(dept);

}

}
// 查询单个

@Test

public void testQueryWrapper2(){

QueryWrapper<Dept> queryWrapper=new QueryWrapper<>();

queryWrapper.eq("deptno", 20);

Dept dept = deptService.getOne(queryWrapper);

System.out.println(dept);

}

update方法:

// 根据 UpdateWrapper 条件

boolean update(Wrapper<T> updateWrapper);

// 根据 whereWrapper 条件,更新记录

boolean update(T updateEntity, Wrapper<T> whereWrapper);

// 根据 ID 选择修改

boolean updateById(T entity);

// 根据ID 批量更新

boolean updateBatchById(Collection<T> entityList);

// 根据ID 批量更新

boolean updateBatchById(Collection<T> entityList, int batchSize);

// 修改

@Test

public void testUpdate(){

// 要更新的数据

Dept dept =new Dept();

dept.setDname("xxx");

dept.setLoc("yyy");

// 更新的条件

QueryWrapper<Dept> queryWrapper=new QueryWrapper<>();

queryWrapper.eq("deptno", 41);

boolean update = deptService.update(dept, queryWrapper);

System.out.println(update);

}

remove方法:

// 根据 entity 条件,删除记录

boolean remove(Wrapper<T> queryWrapper);

// 删除

@Test

public void testRemove(){

QueryWrapper<Dept> queryWrapper=new QueryWrapper<>();

queryWrapper.eq("deptno", 41);

boolean remove = deptService.remove(queryWrapper);

System.out.println(remove);

}

详细文档:CRUD 接口 | MyBatis-Plus

条件构造器

AbstractWrapper

说明: QueryWrapper(LambdaQueryWrapper) 和 UpdateWrapper(LambdaUpdateWrapper) 的父类

用于生成 sql 的 where 条件, entity 属性也用于生成 sql 的 where 条件

注意: entity 生成的 where 条件与 使用各个 api 生成的 where 条件没有任何关联行为

方法 :

  • eq:等于 = 例: eq("name", "老王")--->name = '老王'
  • ne:不等于<> 例: ne("name", "老王")--->name <> '老王'
  • gt:大于> 例: gt("age", 18)--->age > 18
  • ge:大于等于>= 例: ge("age", 18)--->age >= 18
  • lt:小于< 例:lt("age", 18)--->age < 18
  • le:小于等于<= 例:le("age", 18)--->age <= 18
  • between:BETWEEN 值1 AND 值2 例: between("age", 18, 30)--->age between 18 and 30
  • notBetween:NOT BETWEEN 值1 AND 值2 例: notBetween("age", 18, 30)--->age not between 18 and 30
  • like:LIKE '%值%' 例: like("name", "王")--->name like '%王%'
  • notLike:NOT LIKE '%值%' 例: notLike("name", "王")--->name not like '%王%'
  • likeLeft:LIKE '%值' 例: likeLeft("name", "王")--->name like '%王'
  • likeRight:LIKE '值%' 例: likeRight("name", "王")--->name like '王%'
  • isNull:字段 IS NULL 例: isNull("name")--->name is null
  • isNotNull:字段 IS NOT NULL 例: isNotNull("name")--->name is not null
  • in:字段 IN (value.get(0), value.get(1), ...) 例: in("age",{1,2,3})--->age in (1,2,3)
  • notIn:字段 NOT IN (value.get(0), value.get(1), ...) 例: notIn("age",{1,2,3})--->age not in (1,2,3)
  • inSql:字段 IN ( sql语句 ) 例: inSql("age", "1,2,3,4,5,6")--->age in (1,2,3,4,5,6) 例: inSql("id", "select id from table where id < 3")--->id in (select id from table where id < 3)
  • notSql:字段 NOT IN ( sql语句 ) 例: notInSql("age", "1,2,3,4,5,6")--->age not in (1,2,3,4,5,6) 例: notInSql("id", "select id from table where id < 3")--->id not in (select id from table where id < 3)
  • groupBy:分组:GROUP BY 字段, ... 例: groupBy("id", "name")--->group by id,name
  • orderByAse:排序:ORDER BY 字段, ... ASC 例: orderByAsc("id", "name")--->order by id ASC,name ASC
  • orderByDesc:排序:ORDER BY 字段, ... DESC 例: orderByDesc("id", "name")--->order by id DESC,name DESC
  • orderBy:排序:ORDER BY 字段, ... 例: orderBy(true, true, "id", "name")--->order by id ASC,name ASC
  • having:HAVING ( sql语句 ) 例: having("sum(age) > 10")--->having sum(age) > 10 例: having("sum(age) > {0}", 11)--->having sum(age) > 11

or:

拼接 OR

例: eq("id",1).or().eq("name","老王")--->id = 1 or name = '老王'

注意事项: 主动调用or表示紧接着下一个方法不是用and连接!(不调用or则默认为使用and连接)

OR 嵌套

例: or(i -> i.eq("name", "李白").ne("status", "活着"))--->or (name = '李白' and status <> '活着')

and:

AND 嵌套

例: and(i -> i.eq("name", "李白").ne("status", "活着"))--->and (name = '李白' and status <> '活着')

nested:

正常嵌套 不带 AND 或者 OR

例: nested(i -> i.eq("name", "李白").ne("status", "活着"))--->(name = '李白' and status <> '活着')

apply:

拼接 sql

注意事项:

该方法可用于数据库函数 动态入参的params对应前面applySql内部的{index}部分.这样是不会有sql注入风险的,反之会有!

例: apply("id = 1")--->id = 1

例: apply("date_format(dateColumn,'%Y-%m-%d') = '2008-08-08'")--->date_format(dateColumn,'%Y-%m-%d') = '2008-08-08'")

例: apply("date_format(dateColumn,'%Y-%m-%d') = {0}", "2008-08-08")--->date_format(dateColumn,'%Y-%m-%d') = '2008-08-08'")

last:

无视优化规则直接拼接到 sql 的最后

注意事项:

只能调用一次,多次调用以最后一次为准 有sql注入的风险,请谨慎使用

例: last("limit 1")

exists:

拼接 EXISTS ( sql语句 )

例: exists("select id from table where age = 1")--->exists (select id from table where age = 1)

notExists:

拼接 NOT EXISTS ( sql语句 )

例: notExists("select id from table where age = 1")--->not exists (select id from table where age = 1)

select:

设置查询字段

select(String... sqlSelect)

select(Predicate<TableFieldInfo> predicate)

select(Class<T> entityClass, Predicate<TableFieldInfo> predicate)

说明:

以上方法分为两类.

第二类方法为:过滤查询字段(主键除外),入参不包含 class 的调用前需要wrapper内的entity属性有值! 这两类方法重复调用以最后一次为准

例: select("id", "name", "age")

例: select(i -> i.getProperty().startsWith("test"))

UpdateWrapper :

set:

SQL SET 字段

例: set("name", "老李头")

例: set("name", "")--->数据库字段值变为空字符串

例: set("name", null)--->数据库字段值变为null

setSql:

设置 SET 部分 SQL

例: setSql("name = '老李头'")

分页

配置分页插件

@Configuration

@MapperScan("scan.your.mapper.package")

public class MybatisPlusConfig {

/**

* 新的分页插件,一缓和二缓遵循mybatis的规则,需要设置 MybatisConfiguration#useDeprecatedExecutor = false 避免缓存出现问题(该属性会在旧插件移除后一同移除)

*/

@Bean

public MybatisPlusInterceptor mybatisPlusInterceptor() {

MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();

interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.H2));

return interceptor;

}

@Bean

public ConfigurationCustomizer configurationCustomizer() {

return configuration -> configuration.setUseDeprecatedExecutor(false);

}

}

测试分页插件

@Test

public void testPage(){

// 当前页 页大小

QueryWrapper<Dept> queryWrapper=new QueryWrapper<>();

//queryWrapper.likeRight("dname", "A");

Page<Dept> page = deptService.page(new Page<>(1, 2), queryWrapper);

// 当前页数据 总页数 总记录数 当前页 页大小 ... ..

List<Dept> list = page.getRecords();

list.forEach(System.out::println);

System.out.println("总页数:"+page.getPages());

System.out.println("总记录数:"+page.getTotal());

System.out.println("当前页:"+page.getCurrent());

System.out.println("页大小:"+page.getSize());

}

相关推荐

开发者必看的八大Material Design开源项目

MaterialDesign是介于拟物和扁平之间的一种设计风格,自从它发布以来,便引起了很多开发者的关注,在这里小编介绍在Android开发者当中里最受青睐的八个MaterialDesign开源项...

另类插这么可爱,一定是…(另类t恤)

IT之家(www.ithome.com):另类插图:这么可爱,一定是…OSXMavericks和Yosemite打破了苹果对Mac操作系统传统的命名方式,使用加州的某些标志性景点来替换猫...

Android常用ADB命令(安卓adb工具是什么)

杀死应用①根据包名获取APP的PIDadbshellps|grep应用包名②执行kill命令...

微软Mac版PowerPoint测试Reading Order Pane功能

IT之家5月20日消息,微软公司昨日(5月19日)发布博文,邀请Microsoft365Insiders成员,测试macOS新版PowerPoint演示文稿应用,重点引入...

Visual Studio跨平台开发实战(4):Xamarin Android控制项介绍

前言不同于iOS,Xamarin在VisualStudio中针对Android,可以直接设计使用者界面.在本篇教学文章中,笔者会针对Android的专案目录结构以及基本控制项进行介绍,包...

用云存储30分钟快速搭建APP,你信吗?

背景不管你承认与否,移动互联的时代已经到来,这是一个移动互联的时代,手机已经是当今世界上引领潮流的趋势,大型的全球化企业和中小企业都把APP程序开发纳入到他们的企业发展策略当中。但随着手机APP上传的...

谷歌P图神器来了!不用学不用教,输入一句话,分分钟给结果

Pine发自凹非寺量子位|公众号QbitAI当你拍照片时,“模特不好好配合”怎么办?...

iOS文本编辑控件UITextField和UITextVie

记录一个菜鸟的IOS学习之旅,如能帮助正在学习的你,亦枫不胜荣幸;如路过的大神如指教几句,亦枫感激涕淋!细心的朋友可能已经注意到了,IOS学习之旅系列教程在本篇公众号的文章中,封面已经换成美女图片了,...

Android入门图文教程集锦(android 入门教程)

Android入门视频教程集锦AndroidStudio错误gradientandroid:endXattributenotfound...

如何使用Android自定义复合视图(如何使用android自定义复合视图)

在最近的一个客户应用中,我遇到了一个需求,根据选定的值来生成指定数量的编辑框字段,这样用户可以输入人物信息。最初我的想法是把这些逻辑放到Fragment中,只是根据选中值的变化来向线性布局容器中增加编...

原生安卓开发app的框架frida常用关键代码定位

前言有时候可能会对APP进行字符串加密等操作,这样的话你的变量名等一些都被混淆了,看代码就可能无从下手...

教程10 | 三分钟搞定一个智能输入法程序

一案例描述1、考核知识点网格布局线性布局样式和主题Toast2、练习目标掌握网格布局的使用掌握Toast的使用掌握线性布局的使用...

(Android 8.1) 功能与新特性(android的功能)

和你一起终身学习,这里是程序员AndroidAndroid8.1(API级别27)为用户和开发人员引入了各种新特性和功能。本文档重点介绍了开发人员的新功能。通过本章阅读,您将获取到以下内容:Andr...

怎样设置EditText内部文字被锁定不可删除和修改

在做项目的时候,我曾经遇到过这样的要求,就是跟百度贴吧客户端上的一样,在回复帖子的时候,在EditText中显示回复人的名字,而且这个名字不可以修改和删除,说白了就是不可操作,只能在后面输入内容。在E...

如何阻止 Android 活动启动时 EditText 获得焦点

技术背景在Android开发中,当活动启动时,EditText有时会自动获得焦点并弹出虚拟键盘,这可能不是用户期望的行为。为了提升用户体验,我们需要阻止...