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, 则为模块路径中的所有模块。

在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则不一样,需要使用者深入的学习和研究,才能用起来得心应手。

猜你喜欢

转载自www.cnblogs.com/icoolno1/p/13195317.html
今日推荐