ChatGPT解决这个技术问题 Extra ChatGPT

如何从 Jar 运行一个不在其 Manifest 文件中的 Main-Class 的类

我有一个包含 4 个类的 JAR,每个类都有 Main 方法。我希望能够根据需要运行其中的每一个。我正在尝试从 Linux 机器上的命令行运行它。

E.g. The name of my JAR is MyJar.jar

主要类的目录结构如下:

com/mycomp/myproj/dir1/MainClass1.class
com/mycomp/myproj/dir2/MainClass2.class
com/mycomp/myproj/dir3/MainClass3.class
com/mycomp/myproj/dir4/MainClass4.class

我知道我可以在我的清单文件中指定一个类作为主类。但是有什么方法可以让我在命令行上指定一些参数来运行我想运行的任何类?

我试过这个:

jar cfe MyJar.jar com.mycomp.myproj.dir2.MainClass2 com/mycomp/myproj/dir2/MainClass2.class /home/myhome/datasource.properties /home/myhome/input.txt

我得到了这个错误:

com/mycomp/myproj/dir2/MainClass2.class : no such file or directory

(在上述命令中,'/home/myhome/datasource.properties' 和 '/home/myhome/input.txt' 是命令行参数)。

只是将它们打包在不同的 jar 中,使用另一个 jar 来保存依赖项?
为什么没有基于命令行参数调用特定方法(4 个中)的单个主类?

D
Drakes

您可以在其 Manifest 文件中创建没有 Main-Class 的 jar。然后 :

java -cp MyJar.jar com.mycomp.myproj.dir2.MainClass2 /home/myhome/datasource.properties /home/myhome/input.txt

根据文档,这将不起作用:“为了使此选项起作用,JAR 文件的清单必须包含格式为 Main-Class: classname 的行。”
“无法从 MyJar.jar 加载主类清单属性”
托马斯是对的。您需要将“-jar”替换为“-cp”。查看我更新的代码
@chrisdew 提供的答案无需通过修改/删除 Main-Class 属性来更改 Jar 文件。
A
Arun Sai

您可以从 JAR 文件中执行 任何 具有 public static void main 方法的类,即使 jar 文件定义了 Main-Class

执行主类:

java -jar MyJar.jar  // will execute the Main-Class

使用 public static void main 方法执行另一个类:

java -cp MyJar.jar com.mycomp.myproj.AnotherClassWithMainMethod

注意:第一个使用 -jar,第二个使用 -cp


如果你也想要类路径上当前目录中的所有 jars(例如,从 lib 目录运行 Main 类),你可以使用 java -cp "./*" com.mycomp.myproj.AnotherClassWithMainMethod 注意 ./* 周围的双引号,防止 unix 机器扩展它。另请注意,"./*.jar" 不起作用。
Z
Zhou

此答案适用于 Spring-boot 用户:

如果您的 JAR 来自 Spring-boot 项目并使用命令 mvn package spring-boot:repackage 创建,则上述“-cp”方法将不起作用。你会得到:

错误:无法找到或加载主类 your.alternative.class.path

即使您可以通过 jar tvf yours.jar 看到 JAR 中的类。

在这种情况下,请通过以下命令运行您的替代类:

java -cp yours.jar -Dloader.main=your.alternative.class.path org.springframework.boot.loader.PropertiesLauncher

据我了解,Spring-boot 的 org.springframework.boot.loader.PropertiesLauncher 类作为调度入口类,-Dloader.main 参数告诉它要运行什么。

参考:https://github.com/spring-projects/spring-boot/issues/20404


如果你的 jar 是可执行的怎么办。即当你解压jar时,所有的类都以“/BOOT-INF/classes/”开头
它对我有用。您可以在上述命令的末尾将参数作为键/值对
@Zhou,你是救命稻草。谢谢你。
使用 Spring Boot,正是我想要的——非常感谢!
S
Sean Patrick Floyd

除了调用 java -jar myjar.jar com.mycompany.Myclass,您还可以将 Manifest 中的主类设为 Dispatcher 类。

例子:

public class Dispatcher{

    private static final Map<String, Class<?>> ENTRY_POINTS =
        new HashMap<String, Class<?>>();
    static{
        ENTRY_POINTS.put("foo", Foo.class);
        ENTRY_POINTS.put("bar", Bar.class);
        ENTRY_POINTS.put("baz", Baz.class);
    }

    public static void main(final String[] args) throws Exception{

        if(args.length < 1){
            // throw exception, not enough args
        }
        final Class<?> entryPoint = ENTRY_POINTS.get(args[0]);
        if(entryPoint==null){
            // throw exception, entry point doesn't exist
        }
        final String[] argsCopy =
            args.length > 1
                ? Arrays.copyOfRange(args, 1, args.length)
                : new String[0];
        entryPoint.getMethod("main", String[].class).invoke(null,
            (Object) argsCopy);

    }
}

这是一个很好的替代方法!我的jar是使用“spring-boot:repackage”创建的,所以我只是不能通过“-cp”方式运行其他入口类。我不知道 spring-boot 插件对 jar 做了什么。在这种情况下,调度类会有所帮助。
J
JSub

我认为尼克在评论中简要提到的另一个类似选项是创建多个包装罐。我没有尝试过,但我认为除了清单文件之外它们可能是完全空的,清单文件应该指定要加载的主类以及将 MyJar.jar 包含到类路径中。

MyJar1.jar\META-INF\MANIFEST.MF

Manifest-Version: 1.0
Main-Class: com.mycomp.myproj.dir1.MainClass1
Class-Path: MyJar.jar

MyJar2.jar\META-INF\MANIFEST.MF

Manifest-Version: 1.0
Main-Class: com.mycomp.myproj.dir2.MainClass2
Class-Path: MyJar.jar

等等然后用 java -jar MyJar2.jar 运行它


T
Thomas

首先 jar 创建一个 jar,但不运行它。请改用 java -jar

其次,你为什么要通过两次类,作为 FQCN (com.mycomp.myproj.dir2.MainClass2) 和作为文件 (com/mycomp/myproj/dir2/MainClass2.class)?

编辑:

似乎 java -jar 需要指定一个主类。您可以改用 java -cp your.jar com.mycomp.myproj.dir2.MainClass2 ...-cp 在类路径中设置 jar 并使 java 能够在那里查找主类。


从该页面:“它可以在创建或更新 jar 文件时使用。” - 所以这是用于设置主类属性,而不是运行jar。