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

Java -jar "app.jar"

我得到了以下信息:

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

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


当前回答

首先,这有点奇怪,看到你运行java -jar "app"而不是java -jar app.jar

其次,要使jar可执行…你需要jar一个名为META-INF/MANIFEST的文件。曼氏金融

文件本身应该(至少)有这样一行:

Main-Class: com.mypackage.MyClass

其中com.mypackage.MyClass是持有公共静态void main(String[] args)入口点的类。

注意,有几种方法可以用CLI、Maven、Ant或Gradle来完成:

对于CLI,执行以下命令即可:(tks @dvvrt)

jar cmvf META-INF/MANIFEST.MF <new-jar-filename>.jar  <files to include>

对于Maven,像下面这样的代码片段应该可以达到目的。注意,这只是插件定义,而不是完整的pom.xml:

关于这个插件的最新文档:参见https://maven.apache.org/plugins/maven-jar-plugin/

<build>
  <plugins>
    <plugin>
      <!-- Build an executable JAR -->
      <groupId>org.apache.maven.plugins</groupId>
      <artifactId>maven-jar-plugin</artifactId>
      <version>3.1.0</version>
      <configuration>
        <archive>
          <manifest>
            <addClasspath>true</addClasspath>
            <classpathPrefix>lib/</classpathPrefix>
            <mainClass>com.mypackage.MyClass</mainClass>
          </manifest>
        </archive>
      </configuration>
    </plugin>
  </plugins>
</build>

(选择适合您项目的<版本>。)

对于Ant,下面的代码段应该有所帮助:

<jar destfile="build/main/checksites.jar">
  <fileset dir="build/main/classes"/>
  <zipfileset includes="**/*.class" src="lib/main/some.jar"/>
  <manifest>
    <attribute name="Main-Class" value="com.acme.checksites.Main"/>
  </manifest>
</jar>

感谢迈克尔·尼曼德-

Gradle:

plugins {
    id 'java'
}

jar {
    manifest {
        attributes(
                'Main-Class': 'com.mypackage.MyClass'
        )
    }
}

其他回答

任何可执行的jar文件都应该通过点击或使用命令提示符运行,如java -jar app.jar(使用“如果jar路径包含空格”-即java -jar“C:\文件夹名称\app.jar”)。如果您的可执行jar没有运行,这意味着它没有正确创建。

为了更好地理解,提取jar文件(或使用任何工具查看,对于windows 7-Zip是一个很好的工具),并检查/META-INF/MANIFEST.MF下的文件。如果你找到任何条目

Main-Class: your.package.name.ClaaswithMain -那么它就可以了,否则你必须提供它。

注意在MANIFEST上附加Main-Class条目。MF文件,检查你在哪里保存它!

你可以简单地遵循这一步 创建一个jar文件

 jar -cfm jarfile-name manifest-filename Class-file name

在运行jar文件时,像这样简单地运行

 java -cp jarfile-name main-classname

因为您已经添加了MANIFEST。MF,我认为你应该考虑这个文件中Field的顺序。我的env是java版本“1.8.0_91”

还有我的舱单。MF为这里

// MANIFEST.MF
Manifest-Version: 1.0
Created-By: 1.8.0_91 (Oracle Corporation)
Main-Class: HelloWorldSwing

// run
~ java -jar HelloWorldSwing.jar
no main manifest attribute, in HelloWorldSwing.jar

然而,这如下所述

Manifest-Version: 1.0
Main-Class: HelloWorldSwing
Created-By: 1.8.0_91 (Oracle Corporation)

//this run swing normally

以上答案对我来说只是部分帮助。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

对于maven,这就是解决它的方法(对我来说,对于GitHub上的vetle代码库):

<build>
<plugins>
  <plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-shade-plugin</artifactId>
    <version>2.0</version>
    <executions>
      <execution>
        <phase>package</phase>
        <goals>
          <goal>shade</goal>
        </goals>
        <configuration>
          <transformers>
            <transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
              <mainClass>org.lazydevs.veetle.api.VeetleAPI</mainClass>
            </transformer>
          </transformers>
        </configuration>
      </execution>
    </executions>
  </plugin>
 </plugins>
</build>

欢呼声……