zoukankan      html  css  js  c++  java
  • 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}srcmainjava)下加入module-info.java文件。

    module-info.java

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

    再次运行gradlew build出错了:

    E:projects	est1srcmainjavamodule-info.java:3: 错误: 找不到模块: com.google.common
    	requires com.google.common;
    	                   ^
    E:projectssrcmainjavamodule-info.java:4: 错误: 找不到模块: fastjson
    	requires fastjson;
    

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

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

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

    java {
    		modularity.inferModulePath = true
    	}
    

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

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

  • 相关阅读:
    Python 元组
    Python 字典
    Python 列表
    Python 数字(函数)
    Python 序列操作符与函数(字符串)
    JavaScript使用IEEE 标准进行二进制浮点运算,产生莫名错误
    网站用户体验要点(翻译理论)
    去掉checkbox边框的方法
    WPF数据模板样式选择器
    JS中年份问题的纠结!
  • 原文地址:https://www.cnblogs.com/icoolno1/p/13195317.html
Copyright © 2011-2022 走看看