我可以使用类路径覆盖正在运行的jar中的文件吗?

时间:2009-11-10 14:53:13

标签: java jar classpath executable-jar

我有一个JAR文件,其中包含该应用程序的应用程序和配置文件。应用程序从类路径加载配置文件(使用ClassLoader.getResource()),并使用烘焙到JAR文件中的配置文件完全满足其依赖关系。

有时我希望应用程序以稍微不同的配置运行(特别是我想覆盖JDBC URL以指向不同的数据库)所以我创建了一个新的配置文件,将其存储在正确的目录结构中(表示在类路径条目的/config目录中),我想做这样的事情:

java -cp new-config:. -jar application.jar

但是我不能让类路径在应用程序JAR的内容之前有new-config路径条目。它是硬编码的,JAR的内容始终是类路径中的第一件事吗?

4 个答案:

答案 0 :(得分:20)

为什么不在不指定-jar的情况下调用应用程序,而是明确地命名应用程序主类?这将允许您按所需顺序将new-config和application.jar放在类路径中:

e.g。 (假设“new-config”是包含重写属性文件的目录)

java -cp new-config:application.jar Application.Main.Class

我相信主类的名字可以在jar里面的MANIFEST.MF文件中找到....

答案 1 :(得分:13)

使用-jar选项启动应用程序时:

  

... JAR文件是所有文件的来源   用户类和其他用户类   路径设置被忽略

here所述。解决方法是在jar文件的清单中指定类路径以包含附加路径(描述为here)。

但是,鉴于您只是在谈论修改配置,您可能希望采用不依赖于类路径的不同方法。例如,我通常使用属性文件通过Spring配置应用程序以确定数据库的位置等。我的Spring配置在测试,QA和实时环境中是一致的,但是在启动应用程序时我将不同的属性文件作为命令行参数传递

Spring配置代码段

<bean id="MyDataSource" class="org.springframework.jdbc.datasource.SingleConnectionDataSource">
    <property name="url" value="jdbc:microsoft:sqlserver://${dbServer}:${dbPort};DatabaseName=${dbName}"/>
    <property name="username" value="${dbUserName}"/>
    <property name="password" value="${dbPassword}"/>
    <property name="suppressClose" value="false"/>
</bean>

属性文件代码段

dbServer=MyServer
dbPort=1433
dbName=MyDb
dbUserName=Me
dbPassword=foobar

答案 2 :(得分:3)

-jar选项指定的JAR存档会覆盖所有其他值。

您通常必须使用外部配置文件或使用ClassLoader.getResource()构建自己的解决方案。

我们使用自定义解决方案来解决这个问题 - 我们加载内部属性,如下所示:

final Properties p = new Properties();
p.load(DefaultConfiguration.class.getResourceAsStream("config.properties"));

然后我们以相同的方式加载外部文件,并用外部文件覆盖内部值。

有关类加载如何工作的信息,请参阅:

http://java.sun.com/javase/6/docs/technotes/tools/findingclasses.html

答案 3 :(得分:1)

仅使用CLASSPATH可能无法实现。有一些方法可以调用ClassLoader.getResource()使用静态路径来查找资源。如果它正在这样做,它就会绕过CLASSPATH。