自我18年使用 Mybaits 以来,开发环境中如果修改了 xml 文件后,只有重启项目才能生效,如果小项目重启还好,但是对于一个重启需要十几分钟的大型项目来说,这就非常耗时了。开发人员因为修改了xml 文件少量内容,比如添加一个逗号、查询增加一个字段或者修改一个 bug 等,就需要重启整个项目,这就非常痛苦了。

所以在这里给大家推荐一个实现了 Mybatis xml文件热加载的项目,mybatis-xmlreload-spring-boot-starter。它能够帮助我们在Spring Boot + Mybatis的开发环境中修改 xml 后,不需要重启项目就能让修改过后 xml 文件立即生效,实现热加载功能。这里给出项目地址:

ps:mybatis-xmlreload-spring-boot-starter目前 3.0.3.m1 版本实现了 xml 文件修改已有内容,比如修改 sql 语句、添加查询字段、添加查询条件等,可以实现热加载功能。但是对于 xml 文件添加 insert|update|delete|select 标签等内容后,是无法实现热加载的。众所周知,在 Idea 环境进行 Java 开发,在方法内修改方法内容是可以热加载的。但是添加新方法、添加方法参数,修改方法参数,修改方法返回值等都是无法直接热加载的。

一、mybatis-xmlreload-spring-boot-starter使用

mybatis-xmlreload-spring-boot-starter原理:

  • 修改 xml 文件的加载逻辑。在普通的 mybatis-spring 项目中,默认只会加载项目编译过后的 xml 文件,也就是 target 目录下的 xml 文件。但是在mybatis-xmlreload-spring-boot-starter中,修改了这一点,它会加载项目 resources 目录下的 xml 文件,这样用户对于 resources 目录下 xml 文件的修改操作是可以立即触发热加载的。
  • 通过 io.methvin.directory-watcher 项目来监听 xml 文件的修改操作,它底层是通过 java.nio 的WatchService 来实现,当我们监听了整个 resources 目录后,xml 文件的修改会立马触发 MODIFY 事件。
  • 通过 mybatis-spring 项目原生的 xmlMapperBuilder.parse() 方法重新加载解析修改过后的 xml 文件来保证项目对于 Mybatis 的兼容性处理。

二、技术原理

mybatis-xmlreload-spring-boot-starter代码结构如下:



核心代码在MybatisXmlReload类中,执行逻辑:

  1. 通过项目初始化时传入 MybatisXmlReloadProperties prop, List<SqlSessionFactory> sqlSessionFactories 参数,获取mybatis-xmlreload-spring-boot-starter的配置信息,以及项目中的数据源配置
  1. /**
  2. * 是否启动以及xml路径的配置类
  3. */
  4. private MybatisXmlReloadProperties prop;
  5. /**
  6. * 获取项目中初始化完成的SqlSessionFactory列表,对多数据源进行处理
  7. */
  8. private List<SqlSessionFactory> sqlSessionFactories;
  9. public MybatisXmlReload(MybatisXmlReloadProperties prop,
  10. List<SqlSessionFactory> sqlSessionFactories) {
  11. this.prop = prop;
  12. this.sqlSessionFactories = sqlSessionFactories;
  13. }
  1. 解析配置文件指定的 xml 路径,获取 xml 文件在 target 目录下的位置
  1. // 解析项目所有xml路径,获取xml文件在target目录中的位置
  2. List<Resource> mapperLocationsTmp = Stream.of(
  3. Optional.of(prop.getMapperLocations())
  4. .orElse(new String[0]))
  5. .flatMap(location -> Stream.of(getResources(patternResolver, location)))
  6. .toList();
  1. 根据 xml 文件在 target 目录下的位置,进行路径替换找到 xml 文件所在 resources 目录下的位置
  1. // 根据xml文件在target目录下的位置,进行路径替换找到该xml文件在resources目录下的位置
  2. for (Resource mapperLocation : mapperLocationsTmp) {
  3. mapperLocations.add(mapperLocation);
  4. String absolutePath = mapperLocation.getFile().getAbsolutePath();
  5. File tmpFile = new File(absolutePath.replace(CLASS_PATH_TARGET,
  6. MAVEN_RESOURCES));
  7. if (tmpFile.exists()) {
  8. locationPatternSet.add(Path.of(tmpFile.getParent()));
  9. FileSystemResource fileSystemResource =
  10. new FileSystemResource(tmpFile);
  11. mapperLocations.add(fileSystemResource);
  12. }
  13. }
  1. 对 resources 目录的 xml 文件的修改操作进行监听
  1. // 对resources目录的xml文件修改进行监听
  2. List<Path> rootPaths = new ArrayList<>();
  3. rootPaths.addAll(locationPatternSet);
  4. DirectoryWatcher watcher = DirectoryWatcher.builder()
  5. .paths(rootPaths) // or use paths(directoriesToWatch)
  6. .listener(event -> {
  7. switch (event.eventType()) {
  8. case CREATE: /* file created */
  9. break;
  10. case MODIFY: /* file modified */
  11. Path modifyPath = event.path();
  12. String absolutePath = modifyPath.toFile().getAbsolutePath();
  13. logger.info("mybatis xml file has changed:" + modifyPath);
  14. // 执行热加载逻辑...
  15. break;
  16. case DELETE: /* file deleted */
  17. break;
  18. }
  19. })
  20. .build();
  21. ThreadFactory threadFactory = r -> {
  22. Thread thread = new Thread(r);
  23. thread.setName("xml-reload");
  24. thread.setDaemon(true);
  25. return thread;
  26. };
  27. watcher.watchAsync(new ScheduledThreadPoolExecutor(1, threadFactory));
  1. 对多个数据源进行遍历,判断修改过的 xml 文件属于那个数据源
  1. // 对多个数据源进行遍历,判断修改过的xml文件属于那个数据源
  2. for (SqlSessionFactory sqlSessionFactory : sqlSessionFactories) {
  3. ...
  4. }
  1. 根据 Configuration 对象获取对应的标签属性
  1. // 根据 Configuration 对象获取对应的标签属性
  2. Configuration targetConfiguration = sqlSessionFactory.getConfiguration();
  3. Class<?> tClass = targetConfiguration.getClass(),
  4. aClass = targetConfiguration.getClass();
  5. if (targetConfiguration.getClass().getSimpleName()
  6. .equals("MybatisConfiguration")) {
  7. aClass = Configuration.class;
  8. }
  9. Set<String> loadedResources = (Set<String>) getFieldValue(
  10. targetConfiguration, aClass, "loadedResources");
  11. loadedResources.clear();
  12. Map<String, ResultMap> resultMaps = (Map<String, ResultMap>) getFieldValue(
  13. targetConfiguration, tClass, "resultMaps");
  14. Map<String, XNode> sqlFragmentsMaps = (Map<String, XNode>) getFieldValue(
  15. targetConfiguration, tClass, "sqlFragments");
  16. Map<String, MappedStatement> mappedStatementMaps =
  17. (Map<String, MappedStatement>) getFieldValue(
  18. targetConfiguration, tClass, "mappedStatements");
  1. 遍历 resources 目录下 xml 文件列表
  1. // 遍历 resources 目录下 xml 文件列表
  2. for (Resource mapperLocation : mapperLocations) {
  3. ...
  4. }
  1. 判断是否是被修改过的 xml 文件,否则跳过
  1. // 判断是否是被修改过的xml文件,否则跳过
  2. if (!absolutePath.equals(mapperLocation.getFile().getAbsolutePath())) {
  3. continue;
  4. }
  1. 解析xml文件,获取修改后的xml文件标签对应的 resultMaps|sqlFragmentsMaps|mappedStatementMaps 的属性并执行替换逻辑,并且兼容 mybatis-plus 的替换逻辑
  1. // 重新解析xml文件,替换Configuration对象的相对应属性
  2. XPathParser parser = new XPathParser(mapperLocation.getInputStream(),
  3. true,
  4. targetConfiguration.getVariables(),
  5. new XMLMapperEntityResolver());
  6. XNode mapperXnode = parser.evalNode("/mapper");
  7. String namespace = mapperXnode.getStringAttribute("namespace");
  8. List<XNode> resultMapNodes = mapperXnode.evalNodes("/mapper/resultMap");
  9. for (XNode xNode : resultMapNodes) {
  10. String id =
  11. xNode.getStringAttribute("id", xNode.getValueBasedIdentifier());
  12. resultMaps.remove(namespace + "." + id);
  13. }
  14. List<XNode> sqlNodes = mapperXnode.evalNodes("/mapper/sql");
  15. for (XNode sqlNode : sqlNodes) {
  16. String id =
  17. sqlNode.getStringAttribute("id", sqlNode.getValueBasedIdentifier());
  18. sqlFragmentsMaps.remove(namespace + "." + id);
  19. }
  20. List<XNode> msNodes = mapperXnode.evalNodes("select|insert|update|delete");
  21. for (XNode msNode : msNodes) {
  22. String id =
  23. msNode.getStringAttribute("id", msNode.getValueBasedIdentifier());
  24. mappedStatementMaps.remove(namespace + "." + id);
  25. }
  1. 重新加载和解析被修改的 xml 文件
  1. // 9. 重新加载和解析被修改的 xml 文件
  2. try {
  3. XMLMapperBuilder xmlMapperBuilder = new XMLMapperBuilder(
  4. mapperLocation.getInputStream(),
  5. targetConfiguration,
  6. mapperLocation.toString(),
  7. targetConfiguration.getSqlFragments());
  8. xmlMapperBuilder.parse();
  9. } catch (Exception e) {
  10. logger.error(e.getMessage(), e);
  11. }

三、安装方式

  • Spring Boot3.0 中,mybatis-xmlreload-spring-boot-starter在 Maven 项目提供坐标地址如下:
  1. <dependency>
  2. <groupId>com.wayn</groupId>
  3. <artifactId>mybatis-xmlreload-spring-boot-starter</artifactId>
  4. <version>3.0.3.m1</version>
  5. </dependency>
  • Spring Boot2.0 Maven 项目提供坐标地址如下:
  1. <dependency>
  2. <groupId>com.wayn</groupId>
  3. <artifactId>mybatis-xmlreload-spring-boot-starter</artifactId>
  4. <version>2.0.1.m1</version>
  5. </dependency>

四、使用配置

mybatis-xmlreload-spring-boot-starter 目前只有两个配置属性。mybatis-xml-reload.enabled 默认是 false, 也就是不启用 xml 文件的热加载功能,想要开启的话通过在项目配置文件中设置 mybatis-xml-reload.enabled 为 true。还有一个配置属性是 mybatis-xml-reload.mapper-locations,执行热加载的 xml 文件路径,这个属性需要手动填写,跟项目中的 mybatis.mapper-locations 保持一直即可。具体配置如下:

  1. # mybatis xml文件热加载配置
  2. mybatis-xml-reload:
  3. # 是否开启 xml 热更新,true开启,false不开启,默认为false
  4. enabled: true
  5. # xml文件路径,可以填写多个,逗号分隔。
  6. # eg: `classpath*:mapper/**/*Mapper.xml,classpath*:other/**/*Mapper.xml`
  7. mapper-locations: classpath:mapper/*Mapper.xml

五、最后

欢迎大家使用mybatis-xmlreload-spring-boot-starter,这个项目我开源的的,使用中遇到问题可以提交 issue。提交的问题我都会一一查看并回复。再附项目地址:

最后再说一句,感兴趣的朋友可以点赞加关注,你的支持将是我更新动力。

分享一个修改了xml文件再也不用重启的项目mybatis-xmlreload的更多相关文章

  1. 设置tomcat配置文件,在Myeclipse中修改jsp文件之后不用重启tomcat

    在Myeclipse中创建的Web程序在修改类或者jsp页面后需要重动ttomcat的,要重新加载一次的,即重新启动tomcat一次.重启时比较慢,及浪费资源及时间, 设置tomcat配置文件,在My ...

  2. tomcat7修改tomcat-users.xml文件,但服务器重启后又自动还原了。

    tomcat7配置用户管理权限,修改tomcat-users.xml文件 在%tomcat%目录中找到/conf/tomcat-users.xml,修改 <tomcat-users>    ...

  3. 创建新的servlet一定要记得修改web..xml文件!!!

    创建新的servlet一定要记得修改web..xml文件!!!

  4. A query was run and no Result Maps were found for...原来是mapper.xml文件出了问题,是使用MyBatis最常见的一种错误

    今天遇到一个问题,原来是mapper.xml文件出了问题,是使用MyBatis最常见的一种错误 报错的结果是这样的: A query was run and no Result Maps were f ...

  5. Flex air修改外部xml文件 (转)

    AIR的文件操作不难,看完教程应该可以满足你对文件的所有基本操作.这篇教程主要以实际操作中遇到的情况来讲解 我们想想文件操作都会有什么内容,无非是创建,修改,删除,移动,拷贝.在这个过程中我们会涉及到 ...

  6. eclipse配置tomcat后修改server.xml文件(如编码等)无效问题

    我们用eclipse配置好tomcat后,在处理中文乱码或是配置数据源时,我们要修改Tomcat下的server.xml等文件. 修改后重启Tomcat服务器时发现xml文件又被还原了. 因为Tomc ...

  7. 【转】windows7 修改环境变量 和 用不用重启电脑的讨论

      原文:http://www.cnblogs.com/zhenmingliu/archive/2013/02/21/2921396.html   先到我的电脑>属性>高级>环境变量 ...

  8. 【IDEA】IDEA设置修改完JS和JSP不用重启的办法(IDEA热部署)

    修改完JS和JSP不停的重启服务器真的很烦,所以设置修改完之后不用重启也生效: 前提有两个: 确保使用的是debug模式. 确保tomcat是由idea实例化的.也就是说tomcat是在idea中配置 ...

  9. Centos7系统kvm虚机忘记密码进不去, 通过宿主机修改/etc/shadow文件改密码,重启后系统起不来故障排错

    问题描述 某天, 因为其他项目组交接问题, kvm里面的堡垒机系统用户root密码登录不上,然后他通过宿主机修改/etc/shadow文件修改密码,但是修改完后重启系统后发现kvm宿主机连接不上虚机了 ...

  10. xml 文件不给提示(以mybatis 的 mapper映射文件为例)

    在xxx.xml 映射文件的头部可以看到 如下: (mybatis generate 自动生成) <!DOCTYPE mapper PUBLIC "-//mybatis.org//DT ...

随机推荐

  1. 当你的数据集是hdf5格式的文件时,肿么办?

    最近,自己构建了一个卷积神经网络,从网上下载到的数据集是hdf5格式的,希望用这个数据集来训练一下自己构建的这个神经网络. 1. 什么是hdf5? HDF5是二进制数据格式,用于在磁盘上存储巨大的数值 ...

  2. 最短路算法之 Dijkstra

    部分内容参考了李煜东的<算法竞赛进阶指南>,在此声明. 单源最短路径 单源最短路径问题,是说,给定一张有向图(无向图)\(G=(V,E)\) ,\(V\) 是点集,\(E\) 是边集,\( ...

  3. 项目:在wiki标记中添加无序列表(split、join巩固)

    # coding: utf-8 import pyperclip text = pyperclip.paste() lines = text.split("\n") for i i ...

  4. 关于php pconnect长连接如何刷新连接的讨论

    由于每个pconnect所建立的连接信息和单个进程绑定.线上偶发了redis在某一台机器php-fpm上连接正常而无法进行任何操作的问题. 先说结论 查看redis拓展官方文档 close方法 有一句 ...

  5. 【cs231n】knn作业笔记

    完成了assignment-1中knn相关内容的作业,记录一下遇到的知识点和问题 knn.ipynb的内容大致包括: 1.数据集的建立 主要是通过切片函数,如下图选取前5000张图片和其标记作为训练数 ...

  6. 9.22 2020 实验 3:Mininet 实验——测量路径的损耗率

    一.实验目的 在实验 2 的基础上进一步熟悉 Mininet 自定义拓扑脚本,以及与损耗率相关的设定:初步了解 Mininet 安装时自带的 POX 控制器脚本编写,测试路径损耗率.   二.实验任务 ...

  7. webpack 3/4踩坑,我太难了,从安装、卸载、到使用,各相应的版本号,sass-loader报错-版本的原因,webpack -v 不识别,没卸载干净

     -先说卸载: wabpack@4对应的每个插件的版本号都在最后 1 全局安装的话,npm uninstall webpack -g 有时候并不能卸载干净, 2 webpack -v 可判断是否安装成 ...

  8. vue +iview Select省市区联动

    因为需要保存的表里只有City_id一个字段,所以这边只保存"区"的值 <Row type="flex" justify="start" ...

  9. 【MSSQL】AlwaysOn集群增加发布订阅

    在现有AlwaysOn集群增加发布订阅节点 配置 前提 节点1.节点2在AlwaysOn集群,节点3作为集群外节点使用订阅复制集群数据同步 发布对象必须要有主键 步骤 登录节点3配置分发distrib ...

  10. Linux Broadcom Bluetooth BCM43142A0 蓝牙驱动安装

    Linux Broadcom Bluetooth BCM43142A0 蓝牙驱动安装 想转到Linux,奈何蓝牙鼠标不识别. 经历了4个发行版的努力(Linux Mint,Pop!OS,OpenSus ...