我已经安装了一个应用程序,当我试图运行它(它是一个可执行的jar)什么都没有发生。当我从命令行运行它时:

Java -jar "app.jar"

我得到了以下信息:

在“app.jar”中没有主清单属性

通常,如果我自己创建了这个程序,我就会向清单文件添加一个主类属性。但在这种情况下,由于文件来自应用程序,我不能这样做。我还尝试提取jar,看看是否能找到主类,但有很多类,没有一个在它的名称中有“main”这个词。必须有一种方法来修复这个问题,因为程序在其他系统上运行良好。


当前回答

我只想明确一点

Main-Class: <packagename>.<classname>

如果你没有包,你必须忽略这部分,就像这样:

Main-Class: <classname>

其他回答

你可能和我有同样的问题。创建。jar文件后,编写jar xf app.jar META-INF/MANIFEST.MF。这将创建文件的副本到当前目录,以便读取它。如果它只是说:

Manifest-Version: 1.0 创建对象:Oracle Corporation (Oracle Corporation)

并且不包含“Main-Class”声明,那么我认为你找到了你的问题。

但我不知道如何解决这个问题。我在StackOverflow上查询了其他有相同/类似问题的人,但找不到答案。然而,有了这些信息,你可能会得到更好的帮助(考虑到你和我有同样的问题)。

编辑:我尝试了一个清单文件,但没有让它工作,但我的错误是在创建jar文件时只命名了一个类。我写了*.class,现在可以工作了。

虽然我不知道为什么需要创建一个清单文件。但我想只要管用就好。

以上答案对我来说只是部分帮助。java -cp是答案的一部分,但我需要关于如何识别要运行的类的更具体的信息。以下是对我有效的方法:

步骤1:找到我需要运行的类

jar tf /path/to/myjar.jar | more

结果最上面的几行是:

META-INF/
META-INF/MANIFEST.MF
somepath/
somepath/App.class
META-INF/maven/
...

App.class包含了要运行的主类。我不能百分之百确定你是否总是认为你需要的课程是第一个,但它是为我。如果不是,我想使用grep排除与库相关的结果以将类列表缩减到可管理的大小并不太难。

接下来就很简单了:我只需要使用该路径(减去“.class”后缀):

java -cp /path/to/myjar.jar somepath/App

只需将其添加到java模块的build.gradle中。它将创建可执行jar。 它将包含存档中的依赖库。

jar {
  manifest { 
    attributes "Main-Class": "com.company.application.Main"
  }  

  from {
    configurations.compile.collect { it.isDirectory() ? it : zipTree(it) }
  }
}

这将导致[module_name]/build/libs/[module_name].jar文件。 我用shell进行了测试。

Gradle的答案是添加一个jar/manifest/attributes设置,像这样:

apply plugin: 'java'

jar {
    manifest {
        attributes 'Main-Class': 'com.package.app.Class'
    }
}

我个人认为这里所有的答案都是对问题的误解。这个问题的答案在于spring-boot构建.jar的方式不同。每个人都知道Spring Boot设置了一个这样的清单,这与每个人都认为这是一个标准的.jar启动不同,这可能是也可能不是:

Start-Class: com.myco.eventlogging.MyService
Spring-Boot-Classes: BOOT-INF/classes/
Spring-Boot-Lib: BOOT-INF/lib/
Spring-Boot-Version: 1.4.0.RELEASE
Created-By: Apache Maven 3.3.9
Build-Jdk: 1.8.0_131
Main-Class: org.springframework.boot.loader.JarLauncher

也许它需要在类路径上使用org.springframework.boot.loader.JarLauncher来执行?