java launcher_JAR清单类路径不仅适用于Java Application Launcher

java launcher

自从我开始学习Java以来​​,我几乎已经知道, 清单文件中的Class-Path标头字段为可执行JAR (具有由另一个称为Main-Class清单指定应用程序起点的 JAR)指定了相对运行时类路径。 一个同事最近碰到一个让我感到惊讶,因为它证明了一个问题JAR文件的清单的Class-Path条目也影响编译时类路径时包含在类路径中包含JAR在运行javac的 。 这篇文章演示了这种新变化。

Java教程的部署跟踪的“ 将类添加到JAR文件的类路径 ”部分指出:“您指定要包含在applet或应用程序清单文件中的Class-Path头字段中的Class-Path 。” 该部分还指出:“通过使用清单中的Class-Path标头,可以避免在调用Java运行应用程序时指定长的-classpath标志。” 这两个句子从本质上总结了我一直如何想到清单文件中的Class-Path标头:作为包含的JAR的类路径是通过Java应用程序启动器( java可执行文件)执行的。

事实证明,JAR清单中的Class-Path条目会影响Java编译器( javac ),就像它会影响Java应用程序启动器( java )一样。 为了演示这一点,我将使用一个简单的接口( PersonIF ),一个实现该接口的简单类( Person )和一个简单的Main类,该类使用实现该接口的类。 接下来显示这些的代码清单。

PersonIF.java

public interface PersonIF
{void sayHello();
}

人.java

import static java.lang.System.out;public class Person implements PersonIF
{public void sayHello(){out.println("Hello!");}
}

Main.java

public class Main
{public static void main(final String[] arguments){final Person person = new Person();person.sayHello();}
}

从上面的代码清单可以看出, Main类依赖于(使用) Person类,而Person类依赖于(实现) PersonIF 。 我将把PersonIF.class文件放在它自己的名为PersonIF.jar JAR中,并将该JAR存储在(不同的)子目录中。 Person.class文件将存在于其自己的Person.jar JAR文件中,并且该JAR文件包括MANIFEST.MF file ,该MANIFEST.MF fileClass-Path标头引用了相对子目录中的PersonIF.jar

manifestFileInPersonJar参考档案PersonIFJar

现在,我将尝试仅使用类路径中的当前目录从Main.java编译Main.class 。 当javac在单独的子目录中找不到PersonIF.jar时,我以前曾期望编译会失败。 但是,它不会失败!

编译MainJava时没有明确指定PersonIFOnCP

这让我感到惊讶。 当我没有明确指定PersonIF.class (或包含它的JAR)作为通过-cp标志提供的classpath的值时,为什么要编译此文件? 通过使用带有-verbose标志的javac可以看到答案。

compilingMainJavaWoutExplicitSpecifyPersonIFOnCP_verbose

javac -verbose的输出提供“ 文件的搜索路径”和“ 文件的搜索路径”。 在这种情况下,“类文件的搜索路径”很重要,因为我已经将PersonIF.javaPerson.java源文件移到了一个完全不相关的目录中,而不是在那些指定的搜索路径中。 有趣的是,即使我没有在-cp的值中指定此JAR(甚至目录),类文件的搜索路径(以及源文件的搜索路径)仍包含archive/PersonIF.jar 。 这表明Oracle提供的Java编译器考虑在类Class-Path上指定的任何JAR的MANIFEST.MFClass-Path头中指定的类路径内容。

下一个屏幕快照演示了如何运行新编译的Main.class类,以及PersonIF.classarchive/PersonIF.jar获取依赖PersonIF.class archive/PersonIF.jar而无需在传递给Java应用程序启动程序的java -cp标志的值中指定依赖PersonIF.class 。 我希望运行时行为是这种方式,尽管坦白地说我从未尝试过,甚至从未考虑过使用其MANIFEST.MF文件没有Main-Class标头(不可执行的JAR)的JAR来执行此操作。 在此示例中, Person.jar清单文件未指定Main-Class头,而仅指定了Class-Path头,但是在使用java调用时仍能够在运行时使用此类路径内容。

本文的最终演示涉及从JAR文件中删除Class-Path标头和关联的值,并尝试使用javac和相同的命令行指定的classpath进行编译。 在这种情况下,包含Person.class的JAR被称为Person2.jar ,下面的屏幕快照演示了其MANIFEST.MF文件没有Class-Path标头。

Person2JarManifestWithoutClassPathForPersonIF

下一个屏幕快照展示了使用javac现在失败的原因,这是因为,正如预期的那样,没有在类路径上显式指定PersonIF.class ,并且不再通过引用JAR的MANIFEST.MF Class-Path头使它可用。类路径。

notCompilingMainWoutPersonIfInManifestClassPath

从上一个屏幕快照中我们可以看到,源文件和类文件的搜索路径不再包含archive/PersonIF.jar 。 没有可用的JAR, javac将无法找到PersonIF.class并报告错误消息:“找不到PersonIF.class类文件。”

一般观察

  • MANIFEST.MF文件中的Class-Path标头不依赖于存在于同一JAR的MANIFEST.MF文件中的Main-Class标头。
    • 具有Class-Path清单标头的JAR将使这些类路径条目可用于Java类加载器,而不管该JAR是使用java -jar ...执行还是仅放置在较大Java应用程序的类路径上。
  • 因为在JAR清单文件中使用Class-Path的范围并不限于正在执行其Main-Class JAR,所以这些依赖可能潜在地无意中满足了类依赖(即使版本不正确),而不是解析明确指定的classpath条目。 在构造带有指定Class-Path清单的JAR时,或在清单文件中使用带有Class-Path的第三方JAR时,建议谨慎。
  • 有时低估了JAR清单文件的重要性,但是该主题提醒人们了解特定JAR清单文件中的内容的有用性。
  • 本主题提醒您可以不时运行javac而无需-verbose来查看其最新信息即可收集的见解。
  • 每当将JAR放置在javac编译器或java应用程序启动器的类路径上时,您不仅在类路径中的那个JAR中放置了更多的类定义; 您还将在该编译器或应用程序启动器的Class-Path上放置该JAR清单清单的Class-Path引用的所有类和JAR。

结论

Java类加载器可以在许多地方加载用于构建和运行Java应用程序的类。 正如本文所展示的,JAR的MANIFEST.MF文件的Class-Path标头是另一个影响点,可以影响类加载器在运行时和编译时加载的类。 使用Class-Path不会只影响“可执行”的JAR(在清单文件中指定了Main-Class标头,并使用java -jar ...运行),但可能会影响已加载的类以进行编译以及任何Java应用程序执行,其中带有包含Class-Path标头的清单文件的JAR位于类路径上。

翻译自: https://www.javacodegeeks.com/2015/09/jar-manifest-class-path-is-not-for-java-application-launcher-only.html

java launcher

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

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

相关文章

C语言实现银行ATM存取款系统 | 附源码

点击蓝字关注我们来源于网络,侵删银行ATM存取款系统银行ATM存取款系统业务描述如下:银行ATM存取款系统能为用户提供存款、取款、查询、转账和修改密码的功能。为了模拟真实的ATM业务环境,本系统必须实现存款、取款、查询、转账、修改密码以及…

php 链接文件名_7、php-fpm进程管理

1、进程管理php-fpm采用的是master-worker的进程方式。其中,master负责fork worker进程;其次,注册信号,通过信号进行管理worker负责监听端口,等待链接,处理具体的逻辑如下图所示2、信号管理master进程可以理解如下信号…

C语言代码实现平衡二叉树|图解+详细代码

点击蓝字关注我们来源于网络,侵删1. 什么是平衡二叉树平衡二叉树,我们也称【二叉平衡搜索树/AVL】,树中任何节点的两个子树的高度最大差别为1,巴拉巴拉。。。(https://baike.baidu.com/item/AVL树/10986648?fraladdin)但是有个注意的点: 平衡…

cba比赛比分预测_【CBA直播】深圳vs广东前瞻:深圳战广东再掀反攻?

北京时间4月13日晚19点35分,CBA季后赛半决赛第三回合,深圳队主场迎战广东队。尽管目前双方总比分深圳以0-2落后对手,但他们在第二战的顽强表现给人留下了深刻印象。回归主场作战的他们,将在沈梓捷和贺希宁的带领下,力争…

C语言内存泄漏问题及其检视方法

点击蓝字关注我们来源于网络,侵删通过介绍内存泄漏问题原理及检视方法,希望后续能够从编码检视环节就杜绝内存泄漏导致的网上问题发生。本文通过介绍内存泄漏问题原理及检视方法,希望后续能够从编码检视环节就杜绝此类问题发生。预防内存泄漏…

未定义与 struct 类型的输入参数相对应的函数 fetch_引入鲁棒性作为连续参数,这种新损失函数实现了自适应、随时变换...

编辑:陈萍损失函数是机器学习里最基础也是最为关键的一个要素,其用来评价模型的预测值和真实值不一样的程度。最为常见的损失函数包括平方损失、指数损失、log 对数损失等损失函数。这里回顾了一种新的损失函数,通过引入鲁棒性作为连续参数&a…

清理jdk注册表_JDK 9早期版本安装后的Windows注册表清理

清理jdk注册表在我的上一篇博文中 ,我演示了在安装早期版本的JDK 9(内部版本68)之后围绕Oracle Java符号链接 (基于Windows的计算机上的C:\ProgramData\Oracle\Java\javapath\目录)的问题的解决方案。这似乎阻止了早期…

汇编语言调用C语言/C++实例:乘法表

点击蓝字关注我们来源于网络,侵删现在编写一个简单的应用程序,提示用户输入整数,通过移位的方式将其与 2 的幕 (2〜2ⁿ) 相乘,并用填充前导空格的形式再次显示每个乘积。输入-输出使用 C。汇编模块将调用 3 个 C 编写的函数。程序…

rect函数_R函数不会写,quot;抄quot;总会吧!

前面我们简单的介绍了R函数。有些人可能会说,我现在的R水平有限,还不足以写出很高级的函数,该怎么办?俗话说前人栽树后人乘凉,他山之石可以攻玉,鲁迅同志也提出过“拿来”主义。已经有前人,高手…

10个超赞的C语言开源项目,强烈推荐!

点击蓝字关注我们来源于网络,侵删今天分享10个超赞的C语言开源项目,希望这些内容能对大家有所帮助!目录:1. Webbench2. Tinyhttpd3. cJSON4. CMockery5. Libev6. Memcached7. Lua8. SQLite9. UNIX v610. NETBSD1. WebbenchWebbenc…

jboss性能指标_JBoss BRMS复杂事件处理(CEP)性能基准

jboss性能指标技术来了又去,但是一件事保持不变。 在设计企业解决方案时,我们喜欢使我们的生活更轻松的复杂组件,作为建筑师和开发人员,我们一直在寻找使我们的生活更轻松的方法。 一种方法是跟上与感兴趣的技术有关的流行新站点…

C语言经验分享:二维指针与二维数组的两种错误用法

点击蓝字关注我们来源于网络,侵删引子首先看一段代码:void test(int *p) {}int main() {int arr[] {30, 450,14,5};test(arr);return 0; }毫无疑问,上面这段代码是运行OK的。因为C语言标准中有以下规则:在函数参数的声明中,数组名被编译器当作…

camel 使用_使用Camel从WildFly 8向WebLogic 12发送JMS消息

camel 使用系统集成是一个很好的挑战。 特别是当您在寻找通信标准和可靠的解决方案时。 在当今的微服务世界中,每个人都在谈论REST服务和基于http的协议。 实际上,对于大多数通常具有更复杂的需求集的大多数企业项目来说,这是远远不够的。 合…

C++异常处理控制流下的OLLVM混淆

点击蓝字关注我们来源于网络,侵删Inflated!!!C异常化处理OLLVM-控制流平坦化Two PuzzlesException一般碰到C异常逆向,确定了异常分发、处理部分,直接把call throw改为jmp catch块,再F5即可。PS: 多个catch块根据rdx来当为异常处理…

【微服务】springboot整合kafka-stream使用详解

目录 一、前言 二、kafka stream概述 2.1 什么是kafka stream 2.2 为什么需要kafka stream 2.2.1 对接成本低 2.2.2 节省资源 2.2.3 使用简单 2.3 kafka stream特点 2.4 kafka stream中的一些概念 2.5 Kafka Stream应用场景 三、环境准备 3.1 搭建zk 3.1.1 自定义d…

C语言知识总结一:C语言的基本知识汇总

点击蓝字关注我们来源于网络,侵删C语言是一种计算机程序设计语言。它既有高级语言的特点,又具有汇编语言的特点。它可以作 为系统设计语言,编写工作系统应用程序,也可以作为应用程序设计语言,编写不依赖计算机 硬件的应…

jboss8日志级别设置_罐中研讨会:设置JBoss BPM Suite全日研讨会

jboss8日志级别设置是否在寻找一种简单的方法来宣传,展示或演示JBoss业务流程管理套件(BPM Suite)产品的入门难度? 别无所求,因为我们已经召集了这个研讨会,因此您可以围绕JBoss BPM Suite构建一个晚上&a…

yapi 接口文档_1分钟docker部署顶尖 API 文档管理系统

YApi 是高效、易用、功能强大的 api 管理平台,旨在为开发、产品、测试人员提供更优雅的接口管理服务。可以帮助开发者轻松创建、发布、维护 API,YApi 还为用户提供了优秀的交互体验,开发人员只需利用平台提供的接口数据写入工具以及简单的点击…

微软CTO建议业界弃用C/C++采用Rust,遭C++之父回怼

点击蓝字关注我们来源于网络,侵删近日,Microsoft Azure CTO、Sysinternals 的主要开发者 Mark Russinovich 在其社交账号上发布动态称,开发人员是时候停止使用 C/C 来启动新项目,并建议可在需要使用 non-GC 语言的场景中使用 Rust…

红帽 jboss_红帽峰会2015所需的JBoss BPM内容指南

红帽 jboss明年再见? 今年在Red Hat Summit上,我们在JBoss BRMS和JBoss BPM Suite演讲中获得了很多乐趣。 在DevNation周围也有一些社区会议,重点介绍了使我们的产品成为可能的项目。 您可以在他们的博客上找到此演讲的概述,并…