我已经安装了一个应用程序,当我试图运行它(它是一个可执行的jar)什么都没有发生。当我从命令行运行它时:
Java -jar "app.jar"
我得到了以下信息:
在“app.jar”中没有主清单属性
通常,如果我自己创建了这个程序,我就会向清单文件添加一个主类属性。但在这种情况下,由于文件来自应用程序,我不能这样做。我还尝试提取jar,看看是否能找到主类,但有很多类,没有一个在它的名称中有“main”这个词。必须有一种方法来修复这个问题,因为程序在其他系统上运行良好。
我已经安装了一个应用程序,当我试图运行它(它是一个可执行的jar)什么都没有发生。当我从命令行运行它时:
Java -jar "app.jar"
我得到了以下信息:
在“app.jar”中没有主清单属性
通常,如果我自己创建了这个程序,我就会向清单文件添加一个主类属性。但在这种情况下,由于文件来自应用程序,我不能这样做。我还尝试提取jar,看看是否能找到主类,但有很多类,没有一个在它的名称中有“main”这个词。必须有一种方法来修复这个问题,因为程序在其他系统上运行良好。
当前回答
创建文件夹META-INF和文件MANIFEST。MF在该文件夹中,内容如下:
Manifest-Version: 1.0
Class-Path: .
Main-Class: [YOUR_MAIN_CLASS]
然后编译,包括该清单文件。
其他回答
我试过了,对我很有效。 MVN清洁安装包应该工作。
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<executions>
<execution>
<goals>
<goal>repackage</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
</build>
或者,你也可以使用maven-assembly-plugin,如下面的例子所示:
<plugin>
<artifactId>maven-assembly-plugin</artifactId>
<executions>
<execution>
<phase>package</phase>
<goals>
<goal>single</goal>
</goals>
</execution>
</executions>
<configuration>
<archive>
<manifest>
<addClasspath>true</addClasspath>
<mainClass>com.package.MainClass</mainClass>
</manifest>
</archive>
<descriptorRefs>
<descriptorRef>jar-with-dependencies</descriptorRef>
</descriptorRefs>
</configuration>
</plugin>
在这个例子中,节中指定的所有依赖jar都将自动包含在您的单个jar中。注意,jar-with-dependencies应该按字面意思写为,而不是替换为您想要包含的jar文件名。
我刚才得到了同样的错误。 如果你正在使用gradle,只需在你的gradle.build中添加下一个:
apply plugin: 'java'
jar {
manifest {
attributes 'Main-Class': 'com.company.project.MainClass'
}
}
其中com.company.project.MainClass使用公共静态void main(String[] args)方法的类的路径。
以我为例——我从事的是一个多模块项目——我可以用以下方式介绍这个问题:
我将其添加到父文件pom.xml中,这导致了问题。即,值为true的skip:
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<!--
besides hindering the packaging, this also skips running the app after build when calling spring-boot:run. You have to enable it in the
corresponding module by setting skip to false, there.
-->
<skip>true</skip>
</configuration>
</plugin>
</plugins>
</build>
我修复了这个问题,通过添加相同的配置到模块,我想打包成一个jar,但改变了跳过的值为false:
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<version>${spring-boot.version}</version>
<configuration>
<mainClass>${project.mainClass}</mainClass>
<layout>ZIP</layout>
<skip>false</skip>
</configuration>
<executions>
<execution>
<goals>
<goal>repackage</goal>
</goals>
</execution>
<execution>
<id>build-info</id>
<goals>
<goal>build-info</goal>
</goals>
</execution>
</executions>
</plugin>
以上答案对我来说只是部分帮助。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