介绍

POM文件是XML格式的文件,声明性地描述了要使用Maven构建的Java项目的构建结构。 维护大型Java项目的POM XML文件通常很麻烦。 XML是冗长的,POM的结构也需要维护冗余信息。 多次对工件进行命名是多余的,在groupIdartifactId重复名称的某些部分。 如果是多模块项目,则项目的版本应出现在许多文件中。 使用父pom中定义的属性可以减少某些重复,但是您仍然必须在每个模块pom中定义父pom版本,因为您通过工件坐标引用了POM,而不仅仅是将其称为“ pom”。在父目录中”。 依赖项和插件的参数可以在pluginManagementdependency管理的父POM中pluginManagement ,但是尽管每个模块POM通常都是相同的,但您仍然无法摆脱每个模块POM中的插件和依赖项列表。

您可能会与我争论,因为这也是您的口味问题,但是对我来说,XML格式的POM文件太多余了,很难阅读。 也许我不够细致,但是很多时候我都错过了POM文件中的某些错误,并且很难修复它们。

有一些技术可以支持其他格式,但是并未广泛使用。 一种摆脱XML的方法是Poyglot Maven 。 但是,如果您在第一个示例(即Ruby格式的POM)上查看Github页面,您仍然可以看到很多重复的信息。 这是因为Polyglot Maven插入了Maven本身,并且仅将XML格式替换为其他格式,但是无助于POM结构本身的冗余。

在本文中,我将介绍一种比其他解决方案更好的方法,POM文件在构建过程中仍然是XML,因此不需要任何新的插件或更改构建过程,而是使用这些pom.xml文件是使用Jamal宏语言从pom.xml.jam文件以及模块共享的一些其他宏文件生成的。

贾马尔

这个想法是使用基于文本的宏语言从某些源文件生成XML文件,该源文件包含相同的信息(一种简化格式)。 这是某种编程。 宏描述是一个输出详细XML格式的程序。 当宏语言足够强大时,源代码就可以具有足够的描述性,而又不会太冗长。 我的选择是贾马尔。 老实说,选择Jamal的原因之一是它是我大约20年前使用Perl开发的宏语言,而半年前我在Java中重新实现了它。

语言本身非常简单。 文本和宏混合在一起,输出是文本和宏的结果。 宏以{字符或配置的任何其他字符串开头,并以相应的}字符或配置为结束字符串的字符串结尾。 宏可以嵌套,并且可以很好地控制应评估嵌套宏的顺序。 有用户定义的内置宏。 define宏是内置宏之一,用于定义用户定义的宏。

一个例子说得更好。 让我们看一下下面的test.txt.jam文件。

 {@define GAV(_groupId,_artifactId,_version)= { #if |_groupId|<groupId>_groupId</groupId>} { #if |_artifactId|<artifactId>_artifactId</artifactId>} { #if |_version|<version>_version</version>}  }  {GAV :com.javax0.geci:javageci-parent:1.1.2-SNAPSHOT} 

用贾马尔处理它,我们将得到

 <groupId>com.javax0.geci< /groupId > <artifactId>javageci-parent< /artifactId > <version>1.1.2-SNAPSHOT< /version > 

尽管出于排版原因,我还是手动删除了空行,但是您有一个大致的了解。 GAV是使用内置宏define 。 它具有三个名为_groupId_artifactId_version 。 使用宏时,宏主体中的格式参数名称将替换为实际值,并在文本中替换用户定义的宏。 define内置宏本身的文本是一个空字符串。 何时在内置宏前面使用@以及何时使用#有特殊含义,但是在本文中,我无法深入探讨这一细节。

if宏还可以省略groupIdartifactIdversion ,因此

 {GAV :com.javax0.geci:javageci-parent:} 

也可以并且会产生

 <groupId>com.javax0.geci< /groupId > <artifactId>javageci-parent< /artifactId > 

如果您觉得宏的定义中仍然有很多冗余之处,那就对了。 这是定义GAV的简单方法,但是您可以走极端:

 { #define GAV(_groupId,_artifactId,_version)= {@ for z in (groupId,artifactId,version)= { #if |_z|<z>_z</z>} }  }{GAV :com.javax0.geci:javageci-parent:} 

请注意,这需要对宏观评估顺序有一个疯狂的了解,但是作为示例,它表明了这种力量。 关于Jamal的更多信息https://github.com/verhas/jamal

让我们回到原始主题:如何使用Jamal维护POM文件。

烹饪pom果酱

可以有很多方法,每种方法都可能很好。 在这里,我描述了用于Java :: Geci项目的第一种方法。 我创建一个pom.jim文件( jim代表Jamal导入或包含的文件)。 它包含宏的定义,例如GAVdependenciesdependency和许多其他定义。 您可以从Java :: Geci源代码存储库下载该文件: https : pom.jim文件对于所有项目都可以相同,其中没有任何特定的项目。 还有一个version.jim文件,其中包含一个宏,该宏在一个地方定义了项目版本,我在项目中使用的Java版本以及项目的groupId。 当我将发行版号从-SNAPSHOT到下一个发行版或从发行版扩展到下一个-SNAPSHOT这是唯一需要更改它的地方,并且该宏可用于引用顶级POM中的项目版本? 而且在模块POM中引用父对象。

在每个目录中都应该有一个pom.xml文件的地方,我创建了一个pom.xml.jam文件。 该文件导入pom.jim文件,因此可以在其中使用定义的宏。 作为示例,Java :: Geci javageci-engine模块pom.xml.jam文件如下:

 {@ import .. /pom .jim}  {project |jar| {GAV ::javageci-engine:{VERSION}} {parent :javageci-parent} {name|javageci engine} {description|Javageci macro library execution engine} {@include .. /plugins .jim} {dependencies # {@ for MODULE in (api,tools,core)= {dependency :com.javax0.geci:javageci-MODULE:}} {@ for MODULE in (api,engine)= {dependency :org.junit.jupiter:junit-jupiter-MODULE:}} }  } 

我认为这是相当可读的,至少对我而言,它比原始pom.xml更具可读性:

<project xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://maven.apache.org/POM/4.0.0"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"><modelVersion>4.0.0</modelVersion><packaging>jar</packaging><artifactId>javageci-engine</artifactId><version>1.1.1-SNAPSHOT</version><parent><groupId>com.javax0.geci</groupId><artifactId>javageci-parent</artifactId><version>1.1.1-SNAPSHOT</version></parent><name>javageci engine</name><description>Javageci macro library execution engine</description><build><plugins><plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-surefire-plugin</artifactId></plugin><plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-source-plugin</artifactId></plugin><plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-javadoc-plugin</artifactId></plugin></plugins></build><dependencies><dependency><groupId>com.javax0.geci</groupId><artifactId>javageci-api</artifactId></dependency><dependency><groupId>com.javax0.geci</groupId><artifactId>javageci-tools</artifactId></dependency><dependency><groupId>com.javax0.geci</groupId><artifactId>javageci-core</artifactId></dependency><dependency><groupId>org.junit.jupiter</groupId><artifactId>junit-jupiter-api</artifactId></dependency><dependency><groupId>org.junit.jupiter</groupId><artifactId>junit-jupiter-engine</artifactId></dependency></dependencies>
</project>

要启动Jamal,我可以使用Jamal Maven插件。 为此,最简单的方法是在根目录中包含genpom.xml POM文件,其内容为:

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://maven.apache.org/POM/4.0.0"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"><modelVersion>4.0.0</modelVersion><groupId>com.javax0.jamal</groupId><artifactId>pom.xml_files</artifactId><version>out_of_pom.xml.jam_files</version><build><plugins><plugin><groupId>com.javax0.jamal</groupId><artifactId>jamal-maven-plugin</artifactId><version>1.0.2</version><executions><execution><id>execution</id><phase>clean</phase><goals><goal>jamal</goal></goals><configuration><transformFrom>\.jam$</transformFrom><transformTo></transformTo><filePattern>.*pom\.xml\.jam$</filePattern><exclude>target|\.iml$|\.java$|\.xml$</exclude><sourceDirectory>.</sourceDirectory><targetDirectory>.</targetDirectory><macroOpen>{</macroOpen><macroClose>}</macroClose></configuration></execution></executions></plugin></plugins></build>
</project>

有了这个,我可以使用命令行mvn -f genpom.xml clear启动Maven。 这不仅会创建所有POM文件,而且会清除项目的先前编译结果,这在更改POM文件时可能是个好主意。 当目录中还没有pom.xml时,或者由于果酱煮熟的POM文件中可能存在某些错误而导致文件无效时,也可以执行该命令。 不幸的是,所有递归都必须在某处结束,尽管将genpom.xml为果酱熟的POM文件是可行的,但这是不可行的。

摘要

我所描述的是一种使用宏语言作为源而不是原始编辑pom.xml文件的方法。 优点是项目定义更短,更简单。 缺点是需要额外的POM生成步骤,这是手动操作,而不是构建过程的一部分。 您还失去了直接使用Maven版本插件的可能性,因为该插件会修改POM文件。 我自己在使用该插件时总是遇到问题,但这可能是我的错误,而不是该插件的错误。 另外,您必须学习一些Jamal,但是如果您碰巧喜欢它,那也可能是一个优势。 简而言之:如果您愿意,可以尝试一下。 由于该工具(Jamal)已发布在中央genpom.xml ,因此启动非常容易,因此源代码和文档均位于Github上,因此,您所需genpom.xml只是制作genpom.xml文件,煮一些果酱并启动插件。

POM文件不是可以随果酱一起提供的唯一源文件。 我可以轻松想象在产品文档中使用Jamal宏。 您需要做的是创建一个documentationfile.md.jam文件作为源文件,并修改主POM以在将.md.jam转换为生成的宏处理的降价文档的生成过程中运行Jamal。 您也可以像本文中一样设置单独的POM,以防您严格执行转换。 如果您想为Java文件准备一个预处理器,甚至可以拥有java.jam文件,但是我请您不要这样做。 我不想因为给你贾马尔而在地狱里燃烧着永恒的火焰。 不是为了这个目的。

Jamal还有其他许多可能的用途。 它是一种功能强大的宏语言,易于嵌入到应用程序中,并且易于使用Java编写的宏进行扩展。 Java :: Geci还具有一个1.0版本的模块,该模块支持Jamal来简化代码生成,但仍然缺少一些计划通过反射实现Java代码结构的内置宏。 我也在考虑开发一些简单的宏来读取Java源文件并将其包含在文档中。 当我在其中取得一些结果时,我会写。

如果您有任何想法可以使用该技术,请随时与我联系。

翻译自: https://www.javacodegeeks.com/2019/03/get-rid-pom-xml-almost.html

摆脱pom XML…几乎相关推荐

  1. pom.xml中pom全称_摆脱pom XML…几乎

    pom.xml中pom全称 介绍 POM文件是XML格式的文件,声明性地描述了要使用Maven构建的Java项目的构建结构. 维护大型Java项目的POM XML文件通常很麻烦. XML是冗长的,PO ...

  2. POM.xml 标签详解

    pom作为项目对象模型.通过xml表示maven项目,使用pom.xml来实现.主要描述了项目:包括配置文件:开发者需要遵循的规则,缺陷管理系统,组织和licenses,项目的url,项目的依赖性,以 ...

  3. maven POM.xml 标签详解

    pom作为项目对象模型.通过xml表示maven项目,使用pom.xml来实现.主要描述了项目:包括配置文件:开发者需要遵循的规则,缺陷管理系统,组织和licenses,项目的url,项目的依赖性,以 ...

  4. maven项目找不到pom.xml配置的dependency jar中的类问题解决方法

    成功创建一个web项目后,假如我们在web.xml文件中,配置了spring的listener,在pom.xml中也已经作了依赖配置.部署 完项目后,启动项目,发现找不到spring配置的listne ...

  5. Maven使用笔记(四)pom.xml配置详解

    --声明规范 <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3 ...

  6. pom.xml 引入 net.sf.json-lib 报错

    pom.xml 引入 net.sf.json-lib 报错 如下图: 其他的 jar包直接复制过来粘贴都可以使用,没出现啥问题 怎么就他不行 但是: 增加了一个classifier的标签就没有问题了 ...

  7. tomcat8 启动项目pom.xml配置

    tomcat8 启动项目pom.xml配置 <build><finalName>WebTest</finalName><plugins><plug ...

  8. IDEA创建maven项目报错解决:Failed to create a Maven project: 'C:/Users/../IdeaProjects/../pom.xml' already e

    此文首发于我的个人博客:IDEA创建maven项目报错解决 Failed to create a Maven project 'C:/Users/-/IdeaProjects/-/pom.xml' a ...

  9. maven中打包项目为war包的pom.xml配置

    maven中打包成war包的pom.xml配置 (1)完整配置:这个是使用servlet的完整配置,其他的类似. <project xmlns="http://maven.apache ...

最新文章

  1. Android5.1.1源码 - zygote fork出的子进程如何权限降级
  2. Interpreter - 解释器模式
  3. CentOS7.5实践快速部署LAMP+Tomcat成功运行阿里云或者腾讯云
  4. postman中设置关联点
  5. Kubernetes中使用CronJob定时备份etcd集群数据
  6. TensorFlow HOWTO 4.1 多层感知机(分类)
  7. 盘点《令人心动的offer》中让人敲黑板的重点
  8. 海龟绘图两小时上手C语言 - 3 正方形螺旋线
  9. java 笔记(3) —— 动态代理,静态代理,cglib代理
  10. UVa 10791 Minimum Sum LCM
  11. 面试题:React实现鼠标托转文字绕原点旋转
  12. linux 显示目录的大小不一样,linux目录大小显示不实时问题
  13. matlab调和均值滤波_中值和均值滤波---matlab实现
  14. 游戏契合度提示音_产品/市场契合度
  15. TCPIP协议详解----TCP/IP基础知识
  16. android开发环境搭建——android studio
  17. React实战之React+Redux实现一个天气预报小项目
  18. 数据库基础内容(超级详细)
  19. 会导致电脑蓝屏的wav文件原因未知 log whea logger 17 realtek alc269系统播放音频崩溃
  20. 如何在VMware虚拟机中查看Linux的IP地址

热门文章

  1. Codeforces Round #672 (Div. 2)
  2. 【并查集】打击犯罪(ssl 2342)
  3. 线段树-楼房重建-洛谷-P4198
  4. 洛谷P1120小木棒 爆搜+剪枝
  5. HashMap在java并发中如何发生死循环
  6. Intellij Idea乱码解决方案都在这里了
  7. 表扬几位积极的同学!
  8. 《走遍中国》珍藏版(八)
  9. JS生成x到y的随机数
  10. JS对象的属性名规则