Gradle模块化项目中使用了非模块化库的编译方法

引文

Gradle的配置文件有点像Makefile,都是用脚本来控制代码的编译。大体上Gradle跟Maven差不多,因为最终都是把项目文件整理成javac的编译参数,用jar来打包,区别在于形式上的不同,Gradle的编译选项使用的是脚本(Groovy或Kotlin),Maven用的是Xml。据Gradle的官方称,Gradle的编译性能要比Maven快上几倍,刚好最在做一个项目的时候,因为在Maven中加了一个依赖,导致IDE一直处于繁忙状态,只能去任务管理器中把进程结束掉,恢复原来的配置。虽然这个依赖是可有可无的,但促使我有了想尝试一下Gradle的念头。

换成Gradle本身倒不是太难的事情,直接在Maven项目中执行Gradle Init就可以将Maven项目转换成Gradle项目,写了几个测试项目,也没什么问题,但我的项目是跟jfx有关的,这不免要用到java的模块化功能,于是问题就来了。Java的模块化在用Maven构建的时候很正常,一切换到Gradle就不行了,报找不模块的错误。分析了一下,是因为报错的模块都没有经过模块化的,按照Java官方的说法,如果库没有模块化,就放到classpath下,在module-info.java里使用jar包的名称来作为自动模块的模块名。这在Maven下不用特别设置,就可以顺利编译,但在Gradle下边是不行的。大概是Maven作了自动处理,而Gradle没有。翻了无数遍Gradle的文档,终于发现Gradle遇到这种情况确实是需要特别处理的,需要在build.gradle里加一个叫extra-java-module-info的插件,然后使用这个插件申明一下那些未模块化的类库,这样才可以在module-info.java中正常地引入。

以下部分演示如何Gradle项目中使用非模块化(未命名模块unnamed module)库,希望能帮到那些使用Gradle的小伙伴们。

准备工作

安装Gradle工具
由于不是本文的重点,具体过程省略,可以参考以上链接完成Gradle的安装和配置。

生成项目
如何生成项目,参考以上链接。

代码演示

新建一个项目

我创建一个叫test的项目。项目结构跟Maven差不多,只是项目根目录下多了一些跟gradle有关的东西。我们先看一下build.gradle文件,跟构建相关的内容基本上就在这个文件里,初始的build.gradle是个样子:

/*
 * This file was generated by the Gradle 'init' task.
 *
 * This generated file contains a sample Java Library project to get you started.
 * For more details take a look at the Java Libraries chapter in the Gradle
 * User Manual available at https://docs.gradle.org/6.5/userguide/java_library_plugin.html
 */

plugins {
    // Apply the java-library plugin to add support for Java Library
    id 'java-library'
}

repositories {
    // Use jcenter for resolving dependencies.
    // You can declare any Maven/Ivy/file repository here.
	
    jcenter()
}

dependencies {
    // This dependency is exported to consumers, that is to say found on their compile classpath.
    api 'org.apache.commons:commons-math3:3.6.1'

    // This dependency is used internally, and not exposed to consumers on their own compile classpath.
    implementation 'com.google.guava:guava:29.0-jre'

    // Use JUnit test framework
    testImplementation 'junit:junit:4.13'
}

修改仓库

由于墙的原因,默认的仓库下载依赖项会很慢,所以要改国内有镜像,我这里用的阿里Maven仓库镜像。
将:

repositories {
    // Use jcenter for resolving dependencies.
    // You can declare any Maven/Ivy/file repository here.
	
    jcenter()
}

改成:

repositories {
    // Use jcenter for resolving dependencies.
    // You can declare any Maven/Ivy/file repository here.
	maven{
		url 'https://maven.aliyun.com/repository/central'
	}
    mavenCenter()
}

加入依赖包

在dependencies中加入fastjson的依赖包:

dependencies {
    //...省略已有内容
	
    implementation 'com.alibaba:fastjson:1.2.70'

    // ...省略已有内容

}

此时在命令行运行一下gradlew build是没有什么问题的。

模块化

为了实现模块化,需要在源码根目录(${project_path}\src\main\java)下加入module-info.java文件。

module-info.java

module test {
	exports test;
	requires com.google.common;
	requires fastjson;
}

再次运行gradlew build出错了:

E:\projects\test1\src\main\java\module-info.java:3: 错误: 找不到模块: com.google.common
	requires com.google.common;
	                   ^
E:\projects\src\main\java\module-info.java:4: 错误: 找不到模块: fastjson
	requires fastjson;

出现这个问题的原因是没有把依赖项放到module-path里去。在使用javac编译的时候会有这一个参数:

–add-modules <模块>(,<模块>)* 除了初始模块之外要解析的根模块; 如果 为 ALL-MODULE-PATH, 则为模块路径中的所有模块。

extra-java-module-info并没有用到这个编译参数,而是直接修改MANIFEST.MF,或是加入module-info,将未模块的库变成模块化的库。

在gradle中的配置是这样的,用文本编辑器打开build.gradlew,加上这么一段内容:

java {
		modularity.inferModulePath = true
	}

然后我们再次编译,会发现此时错误少了一个。

E:\projects\ebiz\java\test1\src\main\java\module-info.java:4: 错误: 找不到模块: fastjson
	requires fastjson;
	         ^
1 个错误

这其实很好理解,guava-29.0-jre.jar的在MANIFEST.MF文件中指定了Automatic-Module-Name: com.google.common,说明已经是自动模块化了的,所以在模块化项目中直接导入,而fastjson-1.2.70.jar的在MANIFEST.MF文件中没有这一行。这也说明阿里的开发人员并不CARE模块化这玩意,在MANIFEST.MF中加一行Automatic-Module-Name不愿意去做。

配置非模块化库(也叫未命名模块库unnamed module library)

gradle自身不会管你引用的库是不是模块化的,它都统一处理的,为了将非模块化库独立出来,此时我们就要使用一个模块的工具插件了——extra-java-module-info。

我们在plugins里加上一行。

id "de.jjohannes.extra-java-module-info" version "0.1"

然后再加上这一段:

extraJavaModuleInfo {
	// This does not have to be a complete description (e.g. here 'org.apache.commons.collections' does not export anything here).
	// It only needs to be good enough to work in the context of this application we are building.
	module('commons-math3-3.6.1.jar','org.apache.commons','3.6.1')
	module('failureaccess-1.0.1.jar','failureaccess','1.0.1')
	module('jsr305-3.0.2.jar','jsr305','3.0.2')
	module('listenablefuture-9999.0-empty-to-avoid-conflict-with-guava.jar','listenablefuture-9999.0-empty-to-avoid-conflict-with-guava','9999.0')
	module('j2objc-annotations-1.3.jar','j2objc-annotations','1.3')
	module('hamcrest-core-1.3.jar','hamcrest-core','1.3')
	module('fastjson-1.2.70.jar','fastjson','1.2.70') {
		exports("com.alibaba.fastjson")
	}
	automaticModule("guava-29.0-jre.jar", "com.google.common")
}

再次运行gradlew build,结果显示成功。

到这里,在Gradle模块化项目中引入非模块化的演示部分就结束了。

总结

虽然Gradle出来的时候也不短了,但相对Maven来说,用户量也少了不少。由于Maven的广泛使用,该踩的坑都被前人给踩了,使用Maven的时候,可以做到基本上不读文档,完全靠粘贴复制都能混日子,而Gradle则不一样,需要使用者深入的学习和研究,才能用起来得心应手。

注:下文的 *** 代表文件名的组件名称。 # 包含: 文-英文对照文档:【***-javadoc-API文档-文(简体)-英语-对照版.zip】 jar包下载地址:【***.jar下载地址(官方地址+国内镜像地址).txt】 Maven依赖:【***.jar Maven依赖信息(可用于项目pom.xml).txt】 Gradle依赖:【***.jar Gradle依赖信息(可用于项目build.gradle).txt】 源代码下载地址:【***-sources.jar下载地址(官方地址+国内镜像地址).txt】 # 本文件关键字: 文-英文对照文档,英对照文档,java,jar包,Maven,第三方jar包,组件,开源组件,第三方组件,Gradle,文API文档,手册,开发手册,使用手册,参考手册 # 使用方法: 解压 【***.jar文文档.zip】,再解压其的 【***-javadoc-API文档-文(简体)版.zip】,双击 【index.html】 文件,即可用浏览器打开、进行查看。 # 特殊说明: ·本文档为人性化翻译,精心制作,请放心使用。 ·本文档为双语同时展示,一行原文、一行译文,可逐行对照,避免了原文/译文来回切换的麻烦; ·有原文可参照,不再担心翻译偏差误导; ·边学技术、边学英语。 ·只翻译了该翻译的内容,如:注释、说明、描述、用法讲解 等; ·不该翻译的内容保持原样,如:类名、方法名、包名、类型、关键字、代码 等。 # 温馨提示: (1)为了防止解压后路径太长导致浏览器无法打开,推荐在解压时选择“解压到当前文件夹”(放心,自带文件夹,文件不会散落一地); (2)有时,一套Java组件会有多个jar,所以在下载前,请仔细阅读本篇描述,以确保这就是你需要的文件;
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值