02MyBatisPlus条件构造器,自定义SQL,Service接口

news2024/11/19 20:42:51

一、条件构造器

1.MyBatis支持各种复杂的where条件,满足开发的需求

 Wrapper是条件构造器,构建复杂的where查询

 AbstractWrapper有构造where条件的所有方法,QueryWrapper继承后并有自己的select指定查询字段。UpdateWrapper有指定更新的字段的方法

2.案例:基于QueryWrapper查询

①查询出名字带“o”的,存款大于等于1000元的人的id,username,info,balance字段

@Test
void test(){
    // 构建查询条件
    QueryWrapper<User> wrapper = new QueryWrapper<User>()
            .select("id","username","info","balance")
            .like("username","o") // where条件
            .ge("balance",1000);
    // 查询
    user2Mapper.selectList(wrapper);
}

LambdaQueryWrapper方法

@Test
void testl() {
   // 构建查询条件
   LambdaQueryWrapper<User> wrapper = new LambdaQueryWrapper<User>()
         .select(User::getId, User::getUsername, User::getInfo, User::getBalance)
         .like(User::getUsername, "o") // where条件
         .ge(User::getBalance, 1000);
   // 查询
   List<User> users = user2Mapper.selectList(wrapper);
   System.out.println(users);
}

②更新用户名为jack的用户的余额为2000

@Test
void test2(){
    // 更新字段
    User user = new User();
    user.setBalance(2000);
    // 构建查询条件
    QueryWrapper<User> wrapper = new QueryWrapper<User>()
            .eq("username","jack");
    user2Mapper.update(user, wrapper);
}

update(更新的数据,更新的条件)

3.案例:基于UpdateWrapper更新

①更新id为1,2,4的用户的金额扣200

 

@Test
void test3() {
   // 更新条件的ids
   List<Long> ids = new ArrayList<>();
   ids.add(1L);
   ids.add(2L);
   ids.add(4L);
   // 更新的内容和条件
   UpdateWrapper<User> wrapper = new UpdateWrapper<User>()
         .setSql("balance = balance - 200")
         .in("id",ids);
   user2Mapper.update(null, wrapper);
}

4.条件构造器的用法总结

QueryWrapper和LambdaQueryWrapper通常构建select,delete,update的wher条件部分

②UpdateWrapper和LambdaUpdateWrapper通常只有在set语句比较特殊才使用

③尽量使用LambdaQueryWrapper和LambdaUpdateWrapper,避免硬编码。

二、自定义SQL

1.用法

通过MP的Wrapper来构建复杂的where条件,然后自己定义SQL语句剩下的部分。

2.案例将id在指定范围内的用户(1,2,4)的余额扣减指定值

把mp构建好的条件传递到mapper,进行sql组装

①基于Wrapper构建where条件

@Test
void test3() {
   // 更新条件的ids
   List<Long> ids = new ArrayList<>();
   ids.add(1L);
   ids.add(2L);
   ids.add(4L);
   // 更新的内容
   int amount = 200;
   // 编写where更新条件
   LambdaQueryWrapper<User> wrapper = new LambdaQueryWrapper<User>()
         .in(User::getId,ids);
   // 用户自定义SQL
   user2Mapper.updateBalanceByIds(wrapper,amount);
}

②用户自定义mapper方法参数中用Param注解声明wrapper变量,必须是ew

void updateBalanceByIds(@Param("ew") LambdaQueryWrapper<User> wrapper, @Param("amount")int amount);

③编写sql语句,进行where拼接

<update id="updateBalanceByIds">
    update tb_user set balance = balance - #{amount} ${ew.customSqlSegment}
</update>

三、Service接口

1.IService接口的方法

 2.IService接口的继承

 ①自定义Service接口继承IService接口

 ②自定义Service实现类,实现自定义接口并继承ServiceImpl类

 ③新增用户案例

@PostMapping
@ApiOperation("新增")
public void addUser(@RequestBody UserFormDTO userFormDTO){
   // DTO拷贝到PO
   User user = new User();
   BeanUtils.copyProperties(userFormDTO,user);
   // 保存
   userService.save(user);
}

3.案例:IService的Lambda查询

需求:实现一个根据复杂条件查询用户的接口,查询条件如下:

name:用户名关键字,可以为空

status:用户状态,可以为空

原始的MyBatis

 LambdaQuery

@Override
public List<User> getUserList(UserQuery query) {
   // 获取参数
   String name = query.getName();
   Integer status = query.getStatus();
   // 构造查询语句
   return lambdaQuery()
         .like(name != null, User::getUsername, name)
         .eq(status != null, User::getStatus, status)
         .list();
}

4.案例:IService的Lambda更新

需求:改造根据id修改用户余额的接口,要求如下

完成对用户状态校验

完成对用户余额校验

如果扣减后余额为0,则将用户status修改为冻结状态2

@Service
public class UserServiceImpl extends ServiceImpl<User2Mapper, User> implements IUserService {
   @Override
   public void deductBalance(String id, Integer money) {
      // 查询用户
      User user = getById(id);
      // 检验用户状态
      if (user == null || user.getStatus() == 2) {
         throw new RuntimeException("用户状态异常");
      }
      // 检验余额
      if (user.getBalance() < money) {
         throw new RuntimeException("余额不足");
      }
      // 扣减余额
      int remainBalance = user.getBalance() - money;
      // 执行更新
      lambdaUpdate()
            .set(User::getBalance,remainBalance)
            .set(remainBalance==0,User::getStatus,2)
            .eq(User::getId,id)
            .update();
   }

5.IService的批量新增

需求:批量插入10万条用户数据,并作出对比:

  • 普通for循环插入
  • IService的批量插入

①构建批量插入的方法

// 批量插入的方法
private User buildUser(int i){
   User user = new User();
   user.setUsername("Wang_"+i);
   user.setPassword("123");
   user.setPhone("18688990011"+i+"");
   user.setBalance(200);
   user.setInfo("{\"age\": 26, \"intro\": \"英文老师\", \"gender\": \"female\"}");
   user.setCreateTime(LocalDateTime.now());
   user.setUpdateTime(LocalDateTime.now());
   return user;
}

②使用普通的for循环插入

@Test
void testSaveOneByOne() {
   long b = System.currentTimeMillis();
   // 循环插入
   for (int i = 1; i < 100000; i++) {
      userService.save(buildUser(i));
   }
   long e = System.currentTimeMillis();
   System.out.println("耗时" + (e - b));
}

耗时:210005ms,210s

②使用批处理

@Test
void testSaveBatch(){
   // 每次批量插入1000,插入100次就是10万条数据
   // 1.准备1个容量为1000的集合
   List<User> list = new ArrayList<>(1000);
   long b = System.currentTimeMillis();
   for (int i = 0; i < 100000; i++) {
      // 2.添加一个user
      list.add(buildUser(i));
      // 3.每1000条批量插入一次
      if (i % 1000 == 0){
         // 4. 批量插入
         userService.saveBatch(list);
         // 5. 清空
         list.clear();
      }
   }
   long e = System.currentTimeMillis();
   System.out.println("耗时" + (e - b));
}

耗时:26258ms,26s

性能进一步提升,配置jdbc参数,开rewriteBatchedStatements=true

 

总结:

  • 普通for循环逐条插入速度极差,不推荐
  • MP的批量新增,基于预编译的批处理,性能不错
  • 配置jdbc参数,开rewriteBatchedStatements=true,性能最好

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

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

相关文章

bat脚本设置变量有空格踩到的坑

SET PATH c:\xxx;%PATH% 我想把一个路径作为path环境变量最前面的一个&#xff0c;所以使用了上面的语句。 但是没有生效&#xff0c;我还以为是其他什么原因&#xff0c;后来又有一个类似的需求&#xff1a; set output output\x64 结果在使用 %output% 的时候是一个空格&…

常见排序算法之快速排序

快速排序是Hoare于1962年提出的一种二叉树结构的交换排序方法。 基本思想为∶任取待排序元素序列中的某元素作为基准值&#xff0c;按照该排序码将待排序集合分割成两子序列&#xff0c;左子序列中所有元素均小于基准值&#xff0c;右子序列中所有元素均大于基准值&#xff0c;…

Flutter 实战:构建跨平台应用

文章目录 一、简介二、开发环境搭建三、实战案例&#xff1a;开发一个简单的天气应用1. 项目创建2. 界面设计3. 数据获取4. 实现数据获取和处理5. 界面展示6. 添加动态效果和交互7. 添加网络错误处理8. 添加刷新功能9. 添加定位功能10. 添加通知功能11. 添加数据持久化功能 《F…

【Unity细节】Failed importing package???Unity导包失败?

&#x1f468;‍&#x1f4bb;个人主页&#xff1a;元宇宙-秩沅 hallo 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! 本文由 秩沅 原创 &#x1f636;‍&#x1f32b;️收录于专栏&#xff1a;unity细节和bug &#x1f636;‍&#x1f32b;️优质专栏 ⭐【…

计算机服务器中了mallox勒索病毒怎么解决,勒索病毒解密,数据恢复

企业的计算机服务器为企业的数据存储提供了极大便利&#xff0c;也让企业的生产运行效率得到了极大提升&#xff0c;但是网络数据安全威胁随着技术的不断发展也不断增加。近期&#xff0c;云天数据恢复中心接到很多企业的求助&#xff0c;企业的计算机服务器遭到了mallox勒索病…

图神经网络 (GNN)

目录 一、GNN介绍1.1引入1.1.1图的介绍1.1.2怎样将内容表示成图1.1.4图神经网络是在做什么 1.2基本概念 二、GNN流程2.1聚合2.2更新2.2.1一次GNN操作 2.3循环2.3.1多层GNN操作2.3.2能做什么 三、GNN算法原理3.1数据3.2变量定义3.3GNN算法3.3.1Forward3.3.2Backward 四、GNN优势…

Unity 调用自己封装好的DLL库

因为做项目时会用到很多重复的方法&#xff0c;每次都重新写有点浪费时间&#xff0c;就可以将这些方法封装成DLL类库&#xff0c;用的时候直接引入调用就行。 首先在VS里面创建类库文件 注&#xff1a;.NET Framework要选3.5以下 然后定义好命名空间名字和类名就可以写自己要…

互联网按摩预约小程序开发;

随着移动互联网的普及&#xff0c;越来越多的人开始通过手机预约按摩服务。按摩预约小程序是一种方便快捷的预约方式&#xff0c;可以让用户随时随地预约按摩服务。那么&#xff0c;按摩预约小程序的开发周期要多久&#xff1f;按摩预约小程序的功能有哪些呢&#xff1f;本文将…

振南技术干货集:研发版本乱到“妈不认”? Git!(1)

注解目录 1、关于 Git 1.1Git 今生 (Git 和 Linux 的生父都是 Linus&#xff0c;振南给你讲讲当初关于 Git 的爱恨情愁&#xff0c;其背后其实是开源与闭源两左阵营的明争暗斗。) 1.2Git的爆发 (Git 超越时代的分布式思想。振南再给你讲讲旧金山三个年轻人创办 GitHub&…

ABAQUS分析步笔记

定义原则&#xff1a; 每个step的所有边界条件&#xff0c;载荷条件累加构成本step的仿真效果&#xff1b; step2需要在step1的状态基础上进行载荷运动等限定时&#xff0c;需要确保在step2中传递了step1的想要保留的特征&#xff0c;如&#xff1a; 1、BC-1 这里的BC-1的固…

2024最新fl studio 21.2.0.3842中文版完整下载

FL Studio 21.2.0.3842中文版完整下载是最好的音乐开发和制作软件也称为水果音乐软件。它是最受欢迎的工作室&#xff0c;因为它包含了一个主要的听觉工作场所。2024最新fl studioFL Studio 21版有不同的功能&#xff0c;如它包含图形和音乐音序器&#xff0c;帮助您使完美的配…

JRebel热部署——效率提升100倍(程序员工具必备)

1. 下载JRebel 2.激活程序 这里推荐一个免费获取jrebel激活服务器地址和激活邮箱的地址:点击进入 进入网站之后就可以获取到激活链接和邮箱 点击进入激活 复制过去激活就可以 然后就可以看到激活成功了 3.如何使用 代码修改后&#xff0c;直接CtrlShitF9 即可重新启动 4…

阿里云服务器怎么样?阿里云服务器优势、价格及常见问题介绍

阿里云&#xff08;Alibaba Cloud&#xff09;是阿里巴巴集团旗下的云计算服务提供商&#xff0c;其提供的云服务器&#xff08;ECS&#xff09;是其核心服务之一。在云计算市场中&#xff0c;阿里云服务器备受用户的青睐&#xff0c;那么&#xff0c;阿里云服务器究竟怎么样呢…

基于SSM的房屋租售信息管理系统的设计与实现

末尾获取源码 开发语言&#xff1a;Java Java开发工具&#xff1a;JDK1.8 后端框架&#xff1a;SSM 前端&#xff1a;采用JSP技术开发 数据库&#xff1a;MySQL5.7和Navicat管理工具结合 服务器&#xff1a;Tomcat8.5 开发软件&#xff1a;IDEA / Eclipse 是否Maven项目&#x…

“基于RflySim平台飞控底层算法开发”系列专题培训 (第三期)

>> RflySim平台系列专题培训 RflySim平台是一个生态系统或工具链&#xff08;官网&#xff1a;https://doc.rflysim.com&#xff09;&#xff0c;发起于北航可靠飞行控制研究组&#xff0c;主要用于遵循基于模型设计的思想进行无人系统的控制和安全测试。本平台选择MATL…

合成数据加速机器视觉学习

虽然机器学习在基于视觉的自动化中的应用正在增长&#xff0c;但许多行业都面临着挑战&#xff0c;并难以在其计算机视觉应用中实施它。这在很大程度上是由于需要收集许多图像&#xff0c;以及与准确注释这些图像中的不同产品相关的挑战。 该领域的最新趋势之一是利用合成数据…

51单片机PCF8591数字电压表数码管显示设计( proteus仿真+程序+设计报告+讲解视频)

PCF8591数字电压表数码管显示 1.主要功能&#xff1a;讲解视频&#xff1a;2.仿真3. 程序代码4. 设计报告5. 设计资料内容清单&&下载链接资料下载链接&#xff08;可点击&#xff09;&#xff1a; 51单片机PCF8591数字电压表数码管设计( proteus仿真程序设计报告讲解视…

C++以数组作为参数,传递数组地址

文章目录 函数如何使用指针来处理数组将数组作为参数意味着什么数组名和指针对应是好的吗&#xff1f; 参考资料 函数如何使用指针来处理数组 在大多数情况下&#xff0c;C和 C 语言一样&#xff0c;也将数组名视为指针。 C将数组名解释为其第一个元素的地址: cookies &…

YOLOv8 Ultralytics:使用Ultralytics框架训练RT-DETR实时目标检测模型

YOLOv8 Ultralytics&#xff1a;使用Ultralytics框架训练RT-DETR实时目标检测模型 前言相关介绍前提条件实验环境安装环境项目地址LinuxWindows 制作自己的数据集训练自己的数据集创建自己数据集的yaml文件football.yaml文件内容 进行训练进行验证进行预测 数据集获取参考文献 …

NAS 扩容简明指南:使用各种外设给 NAS 们扩容

说起来有趣&#xff0c;NAS 除了“不同设备共享存储”这个功能之外&#xff0c;最重要的功能就是为设备扩容&#xff0c;但是 NAS 自己的存储容量不够了&#xff0c;又该如何。 ​这篇文章分享下我目前使用外设给 NAS 扩容的思路&#xff0c;如何以相对低的成本来获取更大的容…