Gradle入门:依赖管理

即使不是没有可能,创建没有任何外部依赖关系的现实应用程序也是一项挑战。 这就是为什么依赖性管理是每个软件项目中至关重要的部分的原因。

这篇博客文章描述了我们如何使用Gradle管理项目的依赖关系。 我们将学习配置已使用的存储库和所需的依赖项。 我们还将通过实现一个简单的示例应用程序将此理论应用于实践。

让我们开始吧。

补充阅读:

  • Gradle入门:简介可帮助您安装Gradle,描述Gradle构建的基本概念,并描述如何使用Gradle插件向构建中添加功能。
  • Gradle入门:我们的第一个Java项目描述了如何使用Gradle创建Java项目并将应用程序打包到可执行jar文件中。

储存库管理简介

存储库本质上是依赖项容器,每个项目可以使用零个或多个存储库。

Gradle支持以下存储库格式:

  • 常春藤仓库
  • Maven仓库
  • 平面目录存储库

让我们了解如何在构建中配置每种存储库类型。

将常春藤存储库添加到我们的构建中

我们可以使用它的url地址或它在本地文件系统中的位置将Ivy存储库添加到我们的构建中。

如果要使用其常春藤地址添加常春藤存储库,则必须将以下代码片段添加到build.gradle文件中:

repositories {ivy {url "http://ivy.petrikainulainen.net/repo"}
}

如果要使用文件系统中的位置添加常春藤存储库,则必须将以下代码段添加到build.gradle文件中:

repositories {ivy {       url "../ivy-repo"}
}

如果您想获得有关配置Ivy存储库的更多信息,则应检查以下资源:

  • Gradle用户指南的第50.6.6节Ivy存储库
  • IvyArtifactRepository的API文档

让我们继续前进,找出如何将Maven存储库添加到我们的构建中。

将Maven存储库添加到我们的构建中

我们可以使用其URL地址或在本地文件系统中的位置将Maven存储库添加到我们的构建中。

如果要使用其URL添加Maven存储库,则必须将以下代码段添加到build.gradle文件中:

repositories {maven {url "http://maven.petrikainulainen.net/repo"}
}

如果要通过使用文件系统中的Maven存储库来添加Maven存储库,则必须将以下代码段添加到build.gradle文件中:

repositories {maven {       url "../maven-repo"}
}

将Maven存储库添加到构建中时,Gradle可以使用三个“别名”。 这些别名是:

  • mavenCentral()别名意味着从中央Maven 2存储库中获取依赖项。
  • jcenter()别名表示从Bintray的JCenter Maven存储库中获取依赖项。
  • mavenLocal()别名意味着从本地Maven存储库中获取依赖项。

如果要在构建中添加中央Maven 2存储库,则必须将以下代码段添加到build.gradle文件中:

repositories {mavenCentral()
}

如果要获取有关配置Maven存储库的更多信息,则应查阅Gradle用户指南的第50.6.4节“ Maven存储库” 。

让我们继续前进,了解如何向构建中添加平面目录存储库。

将平面目录存储库添加到我们的版本中

如果要使用平面目录存储库,则必须将以下代码片段添加到build.gradle文件中:

repositories {flatDir {dirs 'lib'}
}

这意味着从lib目录中搜索依赖项。 另外,如果需要,可以通过将以下代码段添加到build.gradle文件中来使用多个目录:

repositories {flatDir {dirs 'libA', 'libB'}
}

如果要获取有关平面目录存储库的更多信息,则应检查以下资源:

  • Gradle用户指南的第50.6.5节平面目录存储库
  • Flat Dir Repository发布到gradle用户邮件列表

让我们继续前进,了解如何使用Gradle管理项目的依赖关系。

依赖管理简介

配置项目的存储库后,我们可以声明其依赖项。 如果我们要声明一个新的依赖关系,我们必须执行以下步骤:

  1. 指定依赖项的配置。
  2. 声明所需的依赖项。

让我们仔细看看这些步骤。

将依赖项分组到配置中

在Gradle中,依赖项分为一组命名的依赖项。 这些组称为配置,我们使用它们来声明项目的外部依赖关系。

Java插件指定了几种依赖项配置 ,下面对此进行了描述:

  • 当我们编译项目的源代码时,需要将添加到编译配置的依赖项。
  • 运行时配置包含运行时所需的依赖项。 此配置包含添加到编译配置的依赖项。
  • testCompile配置包含编译项目测试所需的依赖项。 此配置包含我们项目的已编译类以及添加到编译配置中的依赖项。
  • testRuntime配置包含运行我们的测试时所需的依赖项。 此配置包含添加到compileruntimetestCompile配置的依赖项。
  • 档案配置包含由我们的项目产生的工件(例如Jar文件)。
  • 默认配置组包含运行时所需的依赖项。

让我们继续前进,找出如何声明Gradle项目的依赖项。

声明项目的依存关系

最常见的依赖项称为外部依赖项,可从外部存储库找到。 通过使用以下属性来标识外部依赖项:

  • group属性标识依赖项的组(Maven用户将此属性称为groupId )。
  • name属性标识依赖项的名称(Maven用户将此属性称为artifactId )。
  • version属性指定外部依赖项的版本(Maven用户将此属性称为version )。

使用Maven存储库时,这些属性是必需的。 如果使用其他存储库,则某些属性可能是可选的。

例如,如果使用平面目录存储库,则可能仅需要指定name和version 。

假设我们必须声明以下依赖关系:

  • 依赖项的组为“ foo”。
  • 依赖项的名称为“ foo”。
  • 依赖项的版本为0.1。
  • 编译我们的项目时需要依赖项。

我们可以通过将以下代码片段添加到build.gradle文件中来声明此依赖性:

dependencies {compile group: 'foo', name: 'foo', version: '0.1'
}

我们还可以使用遵循以下语法的快捷方式形式声明项目的依赖项: [group]:[name]:[version] 。 如果要使用快捷方式表单,则必须将以下代码片段添加到build.gradle文件中:

dependencies {compile	'foo:foo:0.1'
}

我们还可以将多个依赖项添加到同一配置中。 如果要在声明依赖项时使用“常规”语法,则必须将以下代码片段添加到build.gradle文件中:

dependencies {compile ([group: 'foo', name: 'foo', version: '0.1'],[group: 'bar', name: 'bar', version: '0.1'])
}

另一方面,如果我们要使用快捷方式表单, build.gradle文件的相关部分如下所示:

dependencies {compile 'foo:foo:0.1', 'bar:bar:0.1'
}

自然可以声明属于不同配置的依赖项。 例如,如果我们要声明属于compiletestCompile配置的依赖项,则必须将以下代码片段添加到build.gradle文件中:

dependencies {compile group: 'foo', name: 'foo', version: '0.1'testCompile group: 'test', name: 'test', version: '0.1'
}

同样,可以使用快捷方式表格。 如果要使用快捷方式表单声明相同的依赖项,则build.gradle文件的相关部分如下所示:

dependencies {compile 'foo:foo:0.1'testCompile 'test:test:0.1'
}

阅读第50.4节“如何声明Gradle用户指南”中的依赖项,可以获得有关声明依赖项的更多信息。

现在,我们已经学习了依赖管理的基础知识。 让我们继续并实现示例应用程序。

创建示例应用程序

我们的示例应用程序的要求在以下内容中进行了描述:

  • 示例应用程序的构建脚本必须使用Maven中央存储库。
  • 示例应用程序必须使用Log4j将接收到的消息写入日志。
  • 示例应用程序必须包含单元测试,以确保返回正确的消息。 这些单元测试必须使用JUnit编写。
  • 我们的构建脚本必须创建一个可执行的jar文件。

让我们找出如何满足这些要求。

配置我们的版本库

示例应用程序的要求之一是其构建脚本必须使用Maven中央存储库。 在配置好构建脚本以使用Maven中央存储库之后,其源代码如下所示(相关部分已突出显示):

apply plugin: 'java'repositories {mavenCentral()
}jar {manifest {attributes 'Main-Class': 'net.petrikainulainen.gradle.HelloWorld'}
}

让我们继续并声明示例应用程序的依赖项。

声明示例应用程序的依赖关系

我们必须在build.gradle文件中声明两个依赖

  1. Log4j(版本1.2.17)用于将接收到的消息写入日志。
  2. JUnit(版本4.11)用于为示例应用程序编写单元测试。

声明这些依赖关系后, build.gradle文件如下所示(相关部分突出显示):

apply plugin: 'java'repositories {mavenCentral()
}dependencies {compile 'log4j:log4j:1.2.17'testCompile 'junit:junit:4.11'
}jar {manifest {attributes 'Main-Class': 'net.petrikainulainen.gradle.HelloWorld'}
}

让我们继续写一些代码。

编写代码

为了满足示例应用程序的要求,“我们必须对其进行过度设计”。 我们可以按照以下步骤创建示例应用程序:

  1. 创建一个MessageService类,该类返回字符串“ Hello World!”。 调用其getMessage()方法时。
  2. 创建这确保了MessageServiceTest类的MessageService类返回字符串“世界,你好!”的getMessage()方法。
  3. 创建我们的应用程序的主类,该类从MessageService对象获取消息,并使用Log4j将消息写入日志。
  4. 配置Log4j。

让我们一步一步地完成这些步骤。

首先 ,我们必须在src / main / java / net / petrikainulainen / gradle目录中创建一个MessageService类并实现它。 完成此操作后,其源代码如下所示:

package net.petrikainulainen.gradle;public class MessageService {public String getMessage() {return "Hello World!";}
}

其次 ,我们在src / main / test / net / petrikainulainen / gradle目录中创建了一个MessageServiceTest ,并将单元测试写入了MessageService类的getMessage()方法。 MessageServiceTest类的源代码如下所示:

package net.petrikainulainen.gradle;import org.junit.Before;
import org.junit.Test;import static org.junit.Assert.assertEquals;public class MessageServiceTest {private MessageService messageService;@Beforepublic void setUp() {messageService = new MessageService();}@Testpublic void getMessage_ShouldReturnMessage() {assertEquals("Hello World!", messageService.getMessage());}
}

第三 ,我们在src / main / java / net / petrikainulainen / gradle目录中创建了一个HelloWorld类。 此类是我们应用程序的主要类别。 它从MessageService对象获取消息,然后使用Log4j将其写入日志。 HelloWorld类的源代码如下所示:

package net.petrikainulainen.gradle;import org.apache.log4j.Logger;public class HelloWorld {private static final Logger LOGGER = Logger.getLogger(HelloWorld.class);public static void main(String[] args) {MessageService messageService = new MessageService();String message = messageService.getMessage();LOGGER.info("Received message: " + message);}
}

第四 ,我们必须使用从src / main / resources目录中找到的log4j.properties来配置Log4j。 log4j.properties文件如下所示:

log4j.appender.Stdout=org.apache.log4j.ConsoleAppender
log4j.appender.Stdout.layout=org.apache.log4j.PatternLayout
log4j.appender.Stdout.layout.conversionPattern=%-5p - %-26.26c{1} - %m\nlog4j.rootLogger=DEBUG,Stdout

这就对了。 让我们找出如何运行示例应用程序的测试。

运行单元测试

我们可以使用以下命令运行单元测试:

gradle test

测试通过后,我们将看到以下输出:

> gradle test
:compileJava
:processResources
:classes
:compileTestJava
:processTestResources 
:testClasses
:testBUILD SUCCESSFULTotal time: 4.678 secs

但是,如果我们的单元测试失败,则会看到以下输出(突出显示了有趣的部分):

> gradle test
:compileJava
:processResources
:classes
:compileTestJava
:processTestResources
:testClasses
:testnet.petrikainulainen.gradle.MessageServiceTest > getMessage_ShouldReturnMessageFAILEDorg.junit.ComparisonFailure at MessageServiceTest.java:221 test completed, 1 failed
:test FAILEDFAILURE: Build failed with an exception.* What went wrong:
Execution failed for task ':test'.
> There were failing tests. See the report at: file:///Users/loke/Projects/Java/Blog/gradle-examples/dependency-management/build/reports/tests/index.html* Try:
Run with --stacktrace option to get the stack trace. Run with --info or --debug option to get more log output.BUILD FAILEDTotal time: 4.461 secs

如我们所见,如果我们的单元测试失败,则描述:

  • 哪些测试失败。
  • 运行了多少测试以及失败了多少测试。
  • 测试报告的位置,该报告提供有关失败(和通过)测试的其他信息。

当我们运行单元测试时,Gradle将测试报告创建到以下目录:

  • build / test-results目录包含每个测试运行的原始数据。
  • build / reports / tests目录包含一个HTML报告,该报告描述了我们的测试结果。

HTML测试报告是非常有用的工具,因为它描述了测试失败的原因 。 例如,如果我们的单元测试期望MessageService类的getMessage()方法返回字符串“ Hello Worl1d!”,则该测试用例HTML测试报告将如下所示:

测试失败

让我们继续前进,了解如何打包和运行示例应用程序。

打包并运行我们的示例应用程序

我们可以使用以下命令之一打包应用程序:em> gradle assembly或gradle build 。 这两个命令都会在build / libs目录中创建dependency-management.jar文件。

通过使用命令java -jardependency-management.jar运行示例应用程序时,我们看到以下输出:

> java -jar dependency-management.jarException in thread "main" java.lang.NoClassDefFoundError: org/apache/log4j/Loggerat net.petrikainulainen.gradle.HelloWorld.<clinit>(HelloWorld.java:10)
Caused by: java.lang.ClassNotFoundException: org.apache.log4j.Loggerat java.net.URLClassLoader$1.run(URLClassLoader.java:372)at java.net.URLClassLoader$1.run(URLClassLoader.java:361)at java.security.AccessController.doPrivileged(Native Method)at java.net.URLClassLoader.findClass(URLClassLoader.java:360)at java.lang.ClassLoader.loadClass(ClassLoader.java:424)at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:308)at java.lang.ClassLoader.loadClass(ClassLoader.java:357)... 1 more

发生此异常的原因是,当我们运行应用程序时,没有从类路径中找到Log4j依赖项。

解决此问题的最简单方法是创建一个所谓的“胖” jar文件。 这意味着我们会将所需的依赖项打包到创建的jar文件中。

在按照Gradle Cookbook中的说明进行操作之后,我们的构建脚本如下所示(相关部分已突出显示):

apply plugin: 'java'repositories {mavenCentral()
}dependencies {compile 'log4j:log4j:1.2.17'testCompile 'junit:junit:4.11'
}jar {from { configurations.compile.collect { it.isDirectory() ? it : zipTree(it) } }manifest {attributes 'Main-Class': 'net.petrikainulainen.gradle.HelloWorld'}
}

现在,我们可以运行示例应用程序(打包后),如我们所见,一切都正常运行:

> java -jar dependency-management.jar 
INFO  - HelloWorld                 - Received message: Hello World!

今天就这些。 让我们总结一下我们从此博客文章中学到的知识。

摘要

这篇博客文章教会了我们四件事:

  • 我们学习了如何配置构建所使用的存储库。
  • 我们学习了如何声明所需的依赖关系并将这些依赖关系分组为配置。
  • 我们了解到,Gradle在运行测试时会创建一个HTML测试报告。
  • 我们了解了如何创建一个所谓的“胖” jar文件。

如果您想玩这个博客文章的示例应用程序,可以从Github获得它 。

翻译自: https://www.javacodegeeks.com/2014/07/getting-started-with-gradle-dependency-management.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/363112.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

牛客NOIP暑期七天营-提高组1

牛客NOIP暑期七天营-提高组1 链接 A 边权可为0就排序建一条链子。 但是边权不为0 除了第一个有0的不行。 x连向上一个比他小的数。 期间判断有无解。 #include <bits/stdc.h> #define ll long long using namespace std; const int _2e57; int read() {int x0,f1;char sg…

BZOJ.4009.[HNOI2015]接水果(整体二分 扫描线)

LOJBZOJ洛谷 又是一个三OJ rank1&#xff01;w \(Description\) &#xff08;还是感觉&#xff0c;为啥非要出那种题目背景啊-直接说不好么&#xff09; 给定一棵树和一个路径集合&#xff08;每条路径有一个权值&#xff09;。\(Q\)次询问&#xff0c;每次询问给定一条路径&am…

HTML5常用标签及特殊字符表

*http://html5doctor.com/nav*http://html5doctor.com/article*http://html5doctor.com/section*http://html5doctor.com/asidehttp://html5doctor.com/divhttp://html5doctor.com/figurehttp://html5doctor.com/outlinehttp://html5doctor.com/semantics p 和 span 的理解 p标…

【转载】使用Imaging组件加载GIF动画

Mobil手机加载GIF动态图像的方法有两种&#xff0c;一个就是使用GIF89a标准算法&#xff0c;另一个就是使用SDK自带的Imaging组件&#xff0c;这两种方法是很典型的手机图像处理技术的实践。使用Imaging组件加载GIF比使用标准算法处理高效的多&#xff0c;特别是在处理真彩GIF动…

在Java中获取素数的无限列表

一个常见的问题是确定数字的素因式分解。 蛮力方法是审判部门&#xff08; 维基百科 &#xff0c; 可汗学院 &#xff09;&#xff0c;但是如果必须考虑多个数字&#xff0c;这需要大量的浪费工作。 一种广泛使用的解决方案是Eratosthenes筛&#xff08; 维基百科 &#xff0c…

CF888G XOR-MST trie,贪心

CF888G XOR-MST 链接 CF888G 思路 trie上贪心&#xff0c;先左右两边连边&#xff0c;再用一条边的代价连起左右两颗树。因为内部的边一定比跨两棵树的边权笑&#xff0c;显然是对的。 代码自己瞎yy的。启发式合并 代码 #include <bits/stdc.h> #define ll long long usi…

【处理手记】Configuration system failed to initialize异常的另类原因

有个c#程序在某台电脑上&#xff0c;执行某个操作时&#xff0c;总是会报如图错误&#xff1a; 度娘一番&#xff0c;发现市面上常见的原因是配置文件中的特定节点的位置不对&#xff0c;或者配置文件损坏等等&#xff0c;而这个程序根本没有使用内置的配置文件方案&#xff0c…

学习《Building Applications with FME Objects》 之四 从数据集读取要素

FMEOReader可以访问任何支持格式的数据。 FMEOReader返回两类要素&#xff1a;schema&#xff08;模式&#xff09;要素和数据要素&#xff0c;模式要素用于描述数据集模型。每种支持的格式都有一个模式&#xff0c;一个模式要素是一类要素的数据模型&#xff0c;模式要素描述属…

使用Zapier将应用程序与Neo4j集成

最近&#xff0c;我被带往Zapier &#xff0c;以便在系统之间完成一些轻量级的集成&#xff0c;以快速地进行概念验证。 最初是持怀疑态度的&#xff0c;我发现它确实可以节省时间&#xff0c;并将您从未集成过的系统所有部分捆绑在一起。 而且&#xff0c;这是人们集成他们使…

手机闪屏

表示&#xff0c;本人买了一个华为手机&#xff0c;采用三四个月&#xff0c;就开始闪屏了&#xff0c;手机一划屏就闪&#xff0c;开始的时候表示不能理解&#xff0c;不是手机一般是1年才换吗&#xff1f;突然遇到我也是纠结。 于是乎&#xff0c;浏览了些百度上面的解决方式…

工程能力之C4模型

概述 刚在InfoQ上看到一篇介绍C4Model的文章,觉得这个模型设计的很赞,很有指导意义,做个简单的记录. Why,为什么需要架构图? ThoughtWorks中国 文章中有几句话我觉得很有道理,这里直接摘抄. “纸上的不是架构&#xff0c;每个人脑子里的才是” ; “那些精妙的方案之所以落不了…

[silverlight基础]仿文字连接跑马灯效果-高手绕道

运行效果如下:分析示意图&#xff1a;代码:1<Canvas x:Name"a"Background"AliceBlue"MouseEnter"a_MouseEnter"MouseLeave"a_MouseLeave"Cursor"Hand">2<Canvas.Clip>3<RectangleGeometry RadiusX"0&qu…

Hibernate脏检查的剖析

介绍 持久性上下文使实体状态转换进入队列&#xff0c;该实体状态转换在刷新后转换为数据库语句。 对于托管实体&#xff0c;Hibernate可以代表我们自动检测传入的更改并安排SQL UPDATE。 这种机制称为自动脏检查 。 默认的脏检查策略 默认情况下&#xff0c;Hibernate检查所有…

http GET 和 POST 请求的优缺点、区别以及误区

原文章&#xff1a;https://blog.csdn.net/qq_28483283/article/details/80207674 请优先参考原文章 Get和Post在面试中一般都会问到&#xff0c;一般的区别&#xff1a; &#xff08;1&#xff09;post更安全&#xff08;不会作为url的一部分&#xff0c;不会被缓存、保存在服…

软件工程第二次作业-VSTS单元测试

一、选择开发工具 开发工具选择 Visual studio 2017 社区版&#xff0c;开发语言为C 由于之前已经安装完毕&#xff0c;所以不上传安装过程&#xff0c;主界面如下&#xff1a; 二、练习自动单元测试 使用的测试工具是VSTS&#xff0c;具体步骤如下&#xff1a; 1.编写一个判断…

随便写写(8)

凌晨两点了&#xff0c;还在捣鼓虚拟机&#xff0c;教师节老师们吃不上饭&#xff0c;罪过可就大了。远程有点慢&#xff0c;还遇到个小问题&#xff0c;.netfx 3.5几次都没装上&#xff0c;看了下日志&#xff0c;好像是ghost里的.netfx 2.0有问题&#xff0c;想卸载之&#x…

React后台管理系统-登录页面

登录页面 <div className"col-md-4 col-md-offset-4"> <div className"panel panel-default login-panel"> <div className"panel-heading">欢迎登录 - MMALL管理系统</div> <div className"panel-body"&g…

识别JVM –比预期的要复杂

在Plumbr&#xff0c;我们花了最后一个月的时间来为将来的重大改进奠定基础。 此类构件之一是为JVM添加唯一标识符&#xff0c;以便将来自同一JVM的所有会话链接在一起。 虽然一开始看起来似乎很琐碎的任务&#xff0c;但是当查看JVM捆绑的jps命令的输出时&#xff0c;围绕该问…

随记3

查看本机ip&#xff1a;ifconfig | grep "inet " | grep -v 127.0.0.1去两端空格&#xff1a;rowrow.strip(\t)替换\n \r \t 为空&#xff1a;row row.replace(\r,).replace(\n,).replace(\t,’)字符串长度&#xff1a;aa ‘afebb’ print(len(aa))列表 List列表…

求一架构 : 保留历史数据

最近做一个学校管理系统&#xff0c;当中遇到一个比较麻烦的问题“保留历史数据”&#xff0c;很疑惑&#xff0c;说出来大家讨论下&#xff0c;相信很多人也遇到同样的问题。校方提出需求如下&#xff1a;1.要保留学期留历史数据比如&#xff1a;●  一年级一班2008-2009上学…