MyBatis-Plus 混合使用 XML 和注解

mybatisplus代码生成器:

版本匹配是个比较麻烦的问题,这是我的配置:

       <dependency><groupId>com.baomidou</groupId><artifactId>mybatis-plus-boot-starter</artifactId><version>3.5.2</version></dependency><dependency><groupId>com.baomidou</groupId><artifactId>mybatis-plus-generator</artifactId><version>3.5.2</version></dependency><dependency><groupId>org.freemarker</groupId><artifactId>freemarker</artifactId><version>2.3.31</version> <!-- 推荐版本 --></dependency>
CodeGenerator:
package com.neuedu.hisweb.utils;import com.baomidou.mybatisplus.core.exceptions.MybatisPlusException;
import com.baomidou.mybatisplus.generator.FastAutoGenerator;
import com.baomidou.mybatisplus.generator.config.OutputFile;
import com.baomidou.mybatisplus.generator.config.rules.DateType;
import com.baomidou.mybatisplus.generator.config.rules.NamingStrategy;
import com.baomidou.mybatisplus.generator.engine.FreemarkerTemplateEngine;
import com.baomidou.mybatisplus.generator.fill.Column;
import com.baomidou.mybatisplus.annotation.FieldFill; // 引入正确的枚举类
import org.apache.commons.lang3.StringUtils;import java.util.Collections;
import java.util.Scanner;public class CodeGenerator {private static final String URL = "jdbc:mysql://localhost:3306/his02?useUnicode=true&characterEncoding=utf-8&useSSL=false&allowMultiQueries=true";private static final String USERNAME = "root";private static final String PASSWORD = "1";private static final String AUTHOR = "xj";private static final String PARENT_PACKAGE = "com.neuedu.hisweb";private static final String OUTPUT_DIR = System.getProperty("user.dir") + "/src/main/java";public static String scanner(String tip) {Scanner scanner = new Scanner(System.in);System.out.println("请输入" + tip + ":");if (scanner.hasNext()) {String ipt = scanner.next();if (StringUtils.isNotBlank(ipt)) {return ipt;}}throw new MybatisPlusException("请输入正确的" + tip + "!");}public static void main(String[] args) {// 获取需要生成的表名String tables = scanner("表名,多个英文逗号分割");FastAutoGenerator.create(URL, USERNAME, PASSWORD).globalConfig(builder -> {builder.author(AUTHOR).outputDir(OUTPUT_DIR).fileOverride().disableOpenDir().dateType(DateType.ONLY_DATE).commentDate("yyyy-MM-dd");}).packageConfig(builder -> {builder.parent(PARENT_PACKAGE).entity("entity").service("service").serviceImpl("service.impl").mapper("mapper").controller("controller").pathInfo(Collections.singletonMap(OutputFile.xml,System.getProperty("user.dir") + "/src/main/resources/mapper"));}).strategyConfig(builder -> {builder.addInclude(tables.split(",")).entityBuilder().enableLombok().naming(NamingStrategy.underline_to_camel).columnNaming(NamingStrategy.underline_to_camel).logicDeleteColumnName("deleted").versionColumnName("version").addTableFills(new Column("create_time", FieldFill.INSERT),  // 修改为正确的枚举值new Column("update_time", FieldFill.INSERT_UPDATE)  // 修改为正确的枚举值).controllerBuilder().enableRestStyle().formatFileName("%sController").serviceBuilder().formatServiceFileName("%sService").formatServiceImplFileName("%sServiceImpl").mapperBuilder().formatMapperFileName("%sMapper").formatXmlFileName("%sMapper");}).templateEngine(new FreemarkerTemplateEngine()).execute();}
}

MyBatis-Plus 混合使用 XML 和注解配置

一、配置原则

  1. 优先级:XML 配置会覆盖注解配置。
  2. 最佳实践
    • 简单的 CRUD 操作可以使用注解(如 @Select@Insert 等)。
    • 复杂的 SQL 查询(如多表关联、动态 SQL)建议使用 XML 配置。
  3. XML 文件位置:默认放在 resources/mapper 目录下,文件名与 Mapper 接口名对应。

二、具体配置方法

1. 基础配置

确保 application.yml 或 application.properties 中正确配置 MyBatis-Plus:

# application.yml
mybatis-plus:# Mapper XML 文件路径mapper-locations: classpath:mapper/*.xml# 实体扫描,多个package用逗号或者分号分隔type-aliases-package: com.neuedu.hisweb.entityconfiguration:# 开启驼峰命名map-underscore-to-camel-case: true
2. 混合使用示例

Mapper 接口

package com.neuedu.hisweb.mapper;import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.neuedu.hisweb.entity.User;
import org.apache.ibatis.annotations.*;import java.util.List;public interface UserMapper extends BaseMapper<User> {// 1. 使用注解实现简单查询@Select("SELECT * FROM user WHERE age > #{age}")List<User> selectByAge(int age);// 2. 使用注解实现带参数的插入@Insert("INSERT INTO user(name, age) VALUES(#{name}, #{age})")@Options(useGeneratedKeys = true, keyProperty = "id")int insertUser(User user);// 3. 使用 XML 实现复杂查询(XML 文件中定义该方法)List<User> selectUserWithDepartment();
}

对应的 XML 文件resources/mapper/UserMapper.xml):

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" 
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.neuedu.hisweb.mapper.UserMapper"><!-- 实现 UserMapper 接口中定义的 selectUserWithDepartment 方法 --><select id="selectUserWithDepartment" resultType="com.neuedu.hisweb.entity.User">SELECT u.*, d.name as dept_name FROM user uLEFT JOIN department d ON u.dept_id = d.idWHERE u.deleted = 0</select><!-- 可以添加更多复杂 SQL -->
</mapper>

三、注意事项

  1. 方法名匹配:XML 文件中的 id 必须与 Mapper 接口中的方法名一致。
  2. 参数处理
    • 注解方式:使用 @Param 注解为参数命名。
    • XML 方式:可以直接使用参数名或索引。
  3. 冲突处理
    • 如果同一方法在 XML 和注解中都有定义,XML 配置会覆盖注解配置。
  4. XML 文件路径
    • 确保 mapper-locations 配置正确指向 XML 文件。
    • 推荐使用 classpath:mapper/*.xml 或 classpath*:mapper/**/*.xml

四、动态 SQL 示例

对于复杂的动态 SQL,XML 方式更加灵活:

<select id="selectUserByCondition" resultType="com.neuedu.hisweb.entity.User">SELECT * FROM user<where><if test="name != null and name != ''">AND name LIKE CONCAT('%', #{name}, '%')</if><if test="age != null">AND age &gt;= #{age}</if><if test="deptId != null">AND dept_id = #{deptId}</if></where>ORDER BY create_time DESC
</select>

五、常见问题

  1. XML 文件未被加载

    • 检查 mapper-locations 配置是否正确。
    • 确保 XML 文件在 resources 目录下的正确位置。
  2. 方法找不到

    • 检查 XML 中的 namespace 是否与 Mapper 接口的全限定名一致。
    • 检查方法名(id)是否匹配。
  3. 依赖问题

    • 确保项目中包含 MyBatis-Plus 和 MyBatis 的依赖:

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

@MapperScan 与 mapper-locations 的功能的兼容性

  • @MapperScan 负责让 Spring 容器识别 Mapper 接口。
  • mapper-locations 负责让 MyBatis-Plus 加载 XML 中的 SQL 定义。
    只要两者的路径配置与实际文件位置一致,即可同时使用,不会产生冲突。

一、两者的核心功能区别

  1. @MapperScan("com.neuedu.hisweb.mapper") 的作用

    • 扫描 Mapper 接口:告诉 Spring 容器去哪里查找 MyBatis-Plus 的 Mapper 接口(如 PatientcostsMapper),并将这些接口注册为 Spring Bean。
    • 必须配置:如果没有该注解或未通过其他方式扫描 Mapper 接口,Spring 将无法识别这些接口,导致依赖注入失败(如 @Autowired UserMapper 会报错)。
  2. mapper-locations: classpath:mapper/*.xml 的作用

    • 指定 XML 映射文件路径:告诉 MyBatis-Plus 去哪里查找 XML 格式的 SQL 映射文件(如 PatientcostsMapper.xml)。
    • 可选配置:如果 Mapper 接口中所有方法都使用注解(如 @Select)定义 SQL,则无需配置;若存在 XML 定义的 SQL,则必须配置正确路径。

二、为什么需要同时使用?

  • 场景示例

    • Mapper 接口 PatientcostsMapper 继承自 BaseMapper<Patientcosts>,自动获得 CRUD 方法(无需 XML)。
    • 若该接口中定义了自定义方法(如 selectByCondition),且该方法的 SQL 在 PatientcostsMapper.xml 中实现,则必须通过 mapper-locations 配置 XML 路径,否则 MyBatis-Plus 无法找到对应的 SQL 语句。

  • @MapperScan 解决的是 Mapper 接口的注册问题mapper-locations 解决的是 XML 映射文件的加载问题,两者功能独立,可同时使用。

三、目录结构建议

为避免混淆,建议按以下规范组织文件:

  1. Mapper 接口位置

    src/main/java/com/neuedu/hisweb/mapper/PatientcostsMapper.java
    
     

    (与 @MapperScan("com.neuedu.hisweb.mapper") 扫描路径一致)

  2. XML 映射文件位置

    src/main/resources/mapper/PatientcostsMapper.xml
    
     

    (与 mapper-locations: classpath:mapper/*.xml 配置路径一致)

四、可能的冲突场景与解决方案

1. XML 与注解的 SQL 定义冲突
  • 问题:若 Mapper 接口的某个方法同时通过注解(如 @Select)和 XML 定义 SQL,则 XML 配置优先级更高,会覆盖注解中的 SQL。
  • 解决方案:同一方法的 SQL 定义只能选择一种方式(注解或 XML),避免重复定义。
2. Mapper 接口未被扫描
  • 问题:若 @MapperScan 扫描路径错误(如写成 @MapperScan("com.neuedu.hisweb.dao")),则 Spring 无法找到 PatientcostsMapper,导致启动时报错。
  • 解决方案:确保 @MapperScan 的包路径与 Mapper 接口实际路径一致。

五、验证配置是否生效

  1. 启动应用:若控制台无 NoSuchBeanDefinitionException 或 Invalid bound statement 错误,说明配置正确。
  2. 查看日志
    • 若配置了 mybatis-plus.configuration.log-impl=org.apache.ibatis.logging.stdout.StdOutImpl,执行 SQL 时会打印 XML 或注解中的 SQL 语句,可验证是否正确加载。

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

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

相关文章

基于ssm的教学质量评估系统

博主介绍&#xff1a;java高级开发&#xff0c;从事互联网行业六年&#xff0c;熟悉各种主流语言&#xff0c;精通java、python、php、爬虫、web开发&#xff0c;已经做了六年的毕业设计程序开发&#xff0c;开发过上千套毕业设计程序&#xff0c;没有什么华丽的语言&#xff0…

【STM32】G030单片机开启超过8个ADC通道的方法

如图所示通道数量已经超过8个&#xff0c;按照之前博客的办法已经行不通了 CubeMX配置STM32F103C8T6多路ADC配合DMA采集_stm32f103c8t6的adc采样率-CSDN博客 这里笔者开了10个channel&#xff0c;注意切换为不完全配置&#xff0c;否则的话最多只有8个rank 开DMA&#xff0c;…

不同网络I/O模型的原理

目录 1、I/O的介绍 1.1、I/O 操作分类 1.2、I/O操作流程阶段 1.3、I/O分类 2、同步I/O 2.1、阻塞I/O 2.2、非阻塞I/O 2.3、I/O复用 2.4、信号驱动式I/O 3、异步I/O 前言 在网络I/O之中&#xff0c;I/O操作往往会涉及到两个系统对象&#xff0c;一个是用户空间调用I/O…

在正则表达式中语法 (?P<名字>内容)

&#x1f3af; 重点解释&#xff1a;?P<xxx> 是什么语法&#xff1f; 这一整段&#xff1a; (?P<xxx>...)是 Python 正则表达式中 “命名捕获组” 的语法。 咱们现在一个字一个字来解释&#xff1a; ✅ (?...) 是干啥的&#xff1f; 这是一个捕获组&#xff…

中兴B860AV1.1_MSO9280_降级后开ADB-免刷机破解教程(非刷机)

中兴B860AV1.1江苏移动-自动降级包 关于中兴b860av1.1顽固盒子降级教程终极版 将附件解压好以后&#xff0c;准备一个8G以下的U盘重新格式化为FAT32格式后&#xff0c;并插入电脑 将以下文件及文件夹一同复制到优盘主目录下&#xff08;见下图&#xff09; 全选并复制到U盘主目…

2025-06-13【视频处理】基于视频内容转场进行分割

问题&#xff1a;从网上下载的视频文件&#xff0c;是由很多个各种不同的场景视频片段合并而成。现在要求精确的把各个视频片段从大视频里分割出来。 效果如图&#xff1a;已分割出来的小片段 思考过程 难点在于检测场景变化。为什么呢&#xff1f;因为不同的视频情况各异&am…

ReentrantLock和RLock

文章目录 前言一、 ReentrantLock&#xff08;单机锁&#xff0c;Java 内置&#xff09;示例&#xff1a;方法详解 二、RLock&#xff08;分布式锁&#xff0c;Redisson 提供&#xff09;示例:方法详解 三、 对比总结:四、 如何选择&#xff1f; 前言 ReentrantLock 和 RLock 都…

thinkphp ThinkPHP3.2.3完全开发手册

惯例配置 应用设定 APP_USE_NAMESPACE > true, // 应用类库是否使用命名空间 3.2.1新增 APP_SUB_DOMAIN_DEPLOY > false, // 是否开启子域名部署 APP_SUB_DOMAIN_RULES > array(), // 子域名部署规则 APP_DOMAIN_SUFFIX > , // 域名后缀 如果是…

Python Day50 学习(仍为日志Day19的内容复习)

补充&#xff1a;梳理超参数调整流程&#xff08;逻辑&#xff09; 超参数调节的流程逻辑可以总结为以下几个步骤&#xff1a; 1. 明确目标 确定你要优化的模型和评估指标&#xff08;如准确率、F1值、AUC等&#xff09;。 2. 选择要调节的超参数 列出模型中影响较大的超参数…

公司网络变差的解决方法(固定IP地址冲突)

问题描述 最近公司网络变差&#xff0c;不知道为什么。&#xff08;别的同事反馈的&#xff0c;本人没有感觉变差&#xff0c;也是比较奇怪的现象&#xff09; 现象有视频会议变卡等。 调查过程 1.领导给网络公司打电话沟通&#xff0c;对面远程看了下&#xff0c;不是设备问…

使用Prometheus+Grafana+Alertmanager+Webhook-dingtalk搭建监控平台

一、监控平台介绍 1.监控平台简述普罗米修斯四件套,分别为Prometheus、Grafana、Alertmanager、Webhook-DingTalk。Prometheus一套开源的监控&报警&时间序列数据库的组合,由SoundCloud公司开发,广泛用于云原生环境和容器化应用的监控和性能分析。其提供了通用的数据…

UR机器人解锁关节扭矩控制:利用英伟达Isaac Lab框架,推动装配自动化的Sim2Real迁移

在工业制造领域&#xff0c;机器人装配长期依赖固定自动化模式&#xff0c;面临部署成本高、适配性差等挑战。多部件装配是制造业、汽车及航空航天等行业中的核心环节。传统装配系统通常针对特定任务设计&#xff0c;依赖大量人工工程部署&#xff0c;灵活性不足&#xff0c;难…

ABB 605系列

系列概述 ABB Relion605系列是专为配电网设计的保护继电器产品系列&#xff0c;代表了中低压电力系统保护领域的技术基准。基于ABB在电力保护领域数十年的经验&#xff0c;该系列集成了最新的数字信号处理技术和网络通信能力&#xff0c;为变电站自动化提供了完整的解决方案。…

Python|GIF 解析与构建(6):手搓 tk 录制工具

目录 Python&#xff5c;GIF 解析与构建&#xff08;6&#xff09;&#xff1a;手搓 tk 录制工具 一、工具功能概览 二、核心架构设计 1. 帧率控制模块 2. 屏幕捕获模块 3. 主应用模块 三、关键技术解析 1. 屏幕捕获技术 2. 帧率控制原理 3. 透明窗口实现 四、使用指…

在VBA中,提取word表格的文本时,通常有什么干扰符号,需要清除

标题 在VBA中&#xff0c;提取word表格的文本时&#xff0c;通常有什么干扰符号,需要清除 正文 解决问题提取word表格的文本时&#xff0c;通常有什么干扰符号,需要清除 在VBA中提取Word表格文本时&#xff0c;常见的干扰符号及其清除方法如下&#xff1a; ⚠️ 一、主要干扰符…

C++基础学习:深入理解类中的构造函数、析构函数、this指针与new关键字

前言 在C面向对象编程中&#xff0c;类是构建复杂程序的基本单元。今天&#xff0c;我们将深入探讨类中的几个核心概念&#xff1a;构造函数、析构函数、this指针以及new关键字。这些概念对于理解C对象生命周期和内存管理至关重要。 1. 构造函数 构造函数是类的一个特殊成员…

2025 高考游记/总结

坐标GD 新课标一卷选手 前言 思绪有点乱&#xff0c;想想从哪里说起 没想到这个博客已经三年没发过东西了&#xff0c;上次发还是初三准备特长生的时候&#xff0c;一瞬间就已经高考结束了&#xff0c;有种不真实感 对于高中的三年&#xff0c;有很多话、很多感悟想说&#xff…

Python基础之函数(1/3)

函数(基础) [函数后续还会更新两次] 一.认识函数的作用 函数就是将一段具有独立功能的代码块整合到一个整体并命名&#xff0c;在需要的位置&#xff0c;调用这个名称即可完成对应的需求 函数在开发过程中&#xff0c;可以更高效的实现代码重用 二.函数的使用步骤 1定义函…

AWS CloudFormation实战:构建可复用的ECS服务部署模板

一、前言 在云原生时代,基础设施即代码(IaC)已成为DevOps实践的核心组件。AWS CloudFormation作为AWS原生的IaC服务,允许开发人员和系统管理员以声明式方式定义和部署AWS资源。本文将深入探讨如何构建一个通用的CloudFormation模板,用于在AWS ECS(Elastic Container Servic…

GRUB2 启动配置的工作原理与优先级规则详解

一、核心组件概述 /boot/loader/entries/ 类型:目录,存储 BLS (Boot Loader Specification) 格式的启动项配置文件(如 20-custom-kernel-5.14.0.conf)。管理工具:由 grubby、kernel-install 等工具自动生成或修改。配置内容:每个文件定义一个启动项的详细参数(内核路径、…