34

既存のアプリケーションに Maven jFree 依存関係を追加した後、作成した jar を実行できません。

私が得る唯一のエラーメッセージは次のとおりです。

java -jar target/com.company.product-1.0.0-SNAPSHOT.jar 
Error: Invalid or corrupt jarfile target/com. company.product-1.0.0-SNAPSHOT.jar

完全pom.xmlなものは次のようになります。

<?xml version="1.0" encoding="UTF-8"?>

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
<modelVersion>4.0.0</modelVersion
<groupId>com.mycompany</groupId>
<artifactId>com.mycompany.test</artifactId>
<name>${project.artifactId}</name>
<version>1.0.0-SNAPSHOT</version>

<properties>
    <java-version>1.7</java-version>
    <org.springframework-version>3.1.1.RELEASE</org.springframework-version>
    <org.springframework.data-version>1.0.3.RELEASE</org.springframework.data-version>
    <org.springframework.ws-version>2.0.4.RELEASE</org.springframework.ws-version>
    <org.springframework.ws.oxm-version>1.5.10</org.springframework.ws.oxm-version>
    <org.aspectj-version>1.6.12</org.aspectj-version>
    <org.slf4j-version>1.5.10</org.slf4j-version>
    <selenium-java-version>2.25.0</selenium-java-version>
    <browser-mob-version>2.0-beta-6</browser-mob-version>
</properties>

<dependencies>

    <!-- Hint A: If we delete this dependency it works -->
    <dependency>
        <groupId>org.jfree</groupId>
        <artifactId>jfreechart</artifactId>
        <version>1.0.14</version>
                    <exclusions>
            <exclusion>
                <artifactId>itext</artifactId>
                <groupId>com.lowagie</groupId>
            </exclusion>
        </exclusions>
    </dependency>

    <dependency>
        <groupId>com.itextpdf</groupId>
        <artifactId>itextpdf</artifactId>
        <version>5.3.2</version>
    </dependency>

    <dependency>
        <groupId>de.schlichtherle.io</groupId>
        <artifactId>truezip</artifactId>
        <version>6.6</version>
    </dependency>

    <dependency>
        <groupId>log4j</groupId>
        <artifactId>apache-log4j-extras</artifactId>
        <version>1.1</version>
    </dependency>

    <!-- Caching with ehcache -->
    <dependency>
        <groupId>net.sf.ehcache</groupId>
        <artifactId>ehcache</artifactId>
        <version>2.5.2</version>
        <type>pom</type>
    </dependency>

    <!-- Spring -->
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-context</artifactId>
        <version>${org.springframework-version}</version>
        <exclusions>
            <!-- Exclude Commons Logging in favor of SLF4j -->
            <exclusion>
                <groupId>commons-logging</groupId>
                <artifactId>commons-logging</artifactId>
            </exclusion>
        </exclusions>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-webmvc</artifactId>
        <version>${org.springframework-version}</version>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-core</artifactId>
        <version>${org.springframework-version}</version>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-beans</artifactId>
        <version>${org.springframework-version}</version>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-web</artifactId>
        <version>${org.springframework-version}</version>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-aop</artifactId>
        <version>${org.springframework-version}</version>
    </dependency>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-context-support</artifactId>
        <version>${org.springframework-version}</version>
    </dependency>

    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-tx</artifactId>
        <version>${org.springframework-version}</version>
    </dependency>

    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-orm</artifactId>
        <version>${org.springframework-version}</version>
    </dependency>

    <dependency>
        <groupId>org.springframework.security</groupId>
        <artifactId>spring-security-core</artifactId>
        <version>3.0.3.RELEASE</version>
    </dependency>

    <dependency>
        <groupId>org.springframework.security</groupId>
        <artifactId>spring-security-web</artifactId>
        <version>3.0.3.RELEASE</version>
    </dependency>

    <dependency>
        <groupId>org.springframework.security</groupId>
        <artifactId>spring-security-config</artifactId>
        <version>3.0.3.RELEASE</version>
    </dependency>

    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-test</artifactId>
        <version>${org.springframework-version}</version>
        <scope>test</scope>
    </dependency>
    <!-- Hibernate -->
    <dependency>
        <groupId>org.hibernate.java-persistence</groupId>
        <artifactId>jpa-api</artifactId>
        <version>2.0-cr-1</version>
    </dependency>

    <dependency>
        <groupId>org.hibernate</groupId>
        <artifactId>hibernate-entitymanager</artifactId>
        <version>3.5.1-Final</version>
    </dependency>

    <!-- Database Connectors (HSQL should be removed later) -->
    <dependency>
        <groupId>hsqldb</groupId>
        <artifactId>hsqldb</artifactId>
        <version>1.8.0.7</version>
    </dependency>

    <dependency>
        <groupId>mysql</groupId>
        <artifactId>mysql-connector-java</artifactId>
        <version>5.1.16</version>
    </dependency>

    <!-- AspectJ -->
    <dependency>
        <groupId>org.aspectj</groupId>
        <artifactId>aspectjrt</artifactId>
        <version>${org.aspectj-version}</version>
    </dependency>

    <!-- Logging -->
    <dependency>
        <groupId>org.slf4j</groupId>
        <artifactId>slf4j-api</artifactId>
        <version>${org.slf4j-version}</version>
    </dependency>

    <dependency>
        <groupId>log4j</groupId>
        <artifactId>log4j</artifactId>
        <version>1.2.16</version>
    </dependency>

    <!-- @Inject -->
    <dependency>
        <groupId>javax.inject</groupId>
        <artifactId>javax.inject</artifactId>
        <version>1</version>
    </dependency>

    <dependency>
        <groupId>javax.servlet</groupId>
        <artifactId>jstl</artifactId>
        <version>1.2</version>
    </dependency>

    <!-- Spring Data JPA dependencies -->
    <dependency>
        <groupId>org.springframework.data</groupId>
        <artifactId>spring-data-jpa</artifactId>
        <version>${org.springframework.data-version}</version>
    </dependency>
    <dependency>
        <groupId>javax.ws.rs</groupId>
        <artifactId>jsr311-api</artifactId>
        <version>1.1.1</version>
    </dependency>

    <dependency>
        <groupId>org.springframework.ws</groupId>
        <artifactId>spring-oxm</artifactId>
        <version>${org.springframework.ws.oxm-version}</version>
    </dependency>
    <dependency>
        <groupId>org.springframework.ws</groupId>
        <artifactId>spring-xml</artifactId>
        <version>${org.springframework.ws-version}</version>
    </dependency>
    <dependency>
        <groupId>org.springframework.ws</groupId>
        <artifactId>spring-ws-core</artifactId>
        <version>${org.springframework.ws-version}</version>
    </dependency>
    <dependency>
        <groupId>javax.xml</groupId>
        <artifactId>jaxb-impl</artifactId>
        <version>2.1</version>
    </dependency>
    <dependency>
        <groupId>javax.xml</groupId>
        <artifactId>jaxb-api</artifactId>
        <version>2.1</version>
    </dependency>

    <!-- Test -->
    <dependency>
        <groupId>junit</groupId>
        <artifactId>junit</artifactId>
        <version>4.7</version>
        <scope>test</scope>
    </dependency>

    <dependency>
        <groupId>org.mockito</groupId>
        <artifactId>mockito-all</artifactId>
        <version>1.9.0</version>
        <scope>test</scope>
    </dependency>

    <!-- Common Utils -->
    <dependency>
        <groupId>commons-lang</groupId>
        <artifactId>commons-lang</artifactId>
        <version>2.6</version>
    </dependency>

    <dependency>
        <groupId>commons-collections</groupId>
        <artifactId>commons-collections</artifactId>
        <version>3.2.1</version>
    </dependency>

    <dependency>
        <groupId>commons-cli</groupId>
        <artifactId>commons-cli</artifactId>
        <version>1.2</version>
    </dependency>       

    <!-- Selenium -->
    <dependency>
        <groupId>org.seleniumhq.selenium</groupId>
        <artifactId>selenium-java</artifactId>
        <version>${selenium-java-version}</version>
    </dependency>

    <dependency>
        <groupId>org.seleniumhq.selenium</groupId>
        <artifactId>selenium-firefox-driver</artifactId>
        <version>${selenium-java-version}</version>
    </dependency>

    <dependency>
        <groupId>org.seleniumhq.selenium</groupId>
        <artifactId>selenium-chrome-driver</artifactId>
        <version>${selenium-java-version}</version>
    </dependency>

    <!-- CSV Lib for Keyword Checker -->
    <dependency>
        <groupId>net.sf.opencsv</groupId>
        <artifactId>opencsv</artifactId>
        <version>2.0</version>
    </dependency>

    <!-- Google Places API -->
    <dependency>
        <groupId>com.google.api-client</groupId>
        <artifactId>google-api-client</artifactId>
        <version>1.10.3-beta</version>
        <exclusions>
            <exclusion>
                <artifactId>jackson-core-asl</artifactId>
                <groupId>org.codehaus.jackson</groupId>
            </exclusion>
        </exclusions>
    </dependency>
    <dependency>
        <groupId>com.google.api-client</groupId>
        <artifactId>google-api-client-appengine</artifactId>
        <version>1.10.3-beta</version>
    </dependency>

    <dependency>
        <groupId>org.codehaus.jackson</groupId>
        <artifactId>jackson-mapper-asl</artifactId>
        <version>1.9.9</version>
    </dependency>

    <!-- Google Geocode -->
    <dependency>
        <groupId>com.google.code.geocoder-java</groupId>
        <artifactId>geocoder-java</artifactId>
        <version>0.9</version>
    </dependency>

    <dependency>
        <groupId>com.googlecode.json-simple</groupId>
        <artifactId>json-simple</artifactId>
        <version>1.1</version>
    </dependency>

    <dependency>
        <groupId>com.googlecode.json-simple</groupId>
        <artifactId>json-simple</artifactId>
        <version>1.1</version>
    </dependency>

    <dependency>
        <groupId>net.sf.jgrapht</groupId>
        <artifactId>jgrapht</artifactId>
        <version>0.8.3</version>
    </dependency>

    <dependency>
        <groupId>jgraph</groupId>
        <artifactId>jgraph</artifactId>
        <version>5.13.0.0</version>
    </dependency>

    <!-- Apache Http Client -->
    <dependency>
        <groupId>org.apache.httpcomponents</groupId>
        <artifactId>httpclient</artifactId>
        <version>4.2.1</version>
    </dependency>

    <!-- Amazon web services client -->
    <dependency>
        <groupId>com.amazonaws</groupId>
        <artifactId>aws-java-sdk</artifactId>
        <version>1.2.15</version>
        <exclusions>
            <exclusion>
                <artifactId>jackson-core-asl</artifactId>
                <groupId>org.codehaus.jackson</groupId>
            </exclusion>
        </exclusions>
    </dependency>

    <!-- Docx4j - reading excel files  -->
    <dependency>
        <groupId>org.docx4j</groupId>
        <artifactId>docx4j</artifactId>
        <version>2.8.0</version>
    </dependency>

    <!-- Browser Mob Proxy -->

    <dependency>
        <groupId>biz.neustar</groupId>
        <artifactId>browsermob-proxy</artifactId>
        <version>${browser-mob-version}</version>
        <exclusions>
            <exclusion>
                <groupId>org.seleniumhq.selenium</groupId>
                <artifactId>selenium-api</artifactId>
            </exclusion>
            <exclusion>
                <artifactId>icu4j</artifactId>
                <groupId>com.ibm.icu</groupId>
            </exclusion>
            <exclusion>
                <artifactId>jackson-mapper-asl</artifactId>
                <groupId>org.codehaus.jackson</groupId>
            </exclusion>
            <exclusion>
                <artifactId>jackson-core-asl</artifactId>
                <groupId>org.codehaus.jackson</groupId>
            </exclusion>
            <exclusion>
                <groupId>org.slf4j</groupId>
                <artifactId>slf4j-jdk14</artifactId>
            </exclusion>
        </exclusions>
    </dependency>

    <!-- Hint B: If we copy this Apache POI dependencies to the top, it works -->
    <!-- Apache POI - for reading xlsx files -->
    <dependency>
        <groupId>org.apache.poi</groupId>
        <artifactId>poi</artifactId>
        <version>3.8</version>
    </dependency>

    <dependency>
        <groupId>org.apache.poi</groupId>
        <artifactId>poi-ooxml</artifactId>
        <version>3.8</version>
    </dependency>

    <dependency>
        <groupId>org.apache.xmlbeans</groupId>
        <artifactId>xmlbeans</artifactId>
        <version>2.5.0</version>
    </dependency>

    <dependency>
        <groupId>org.apache.poi</groupId>
        <artifactId>ooxml-schemas</artifactId>
        <version>1.1</version>
    </dependency>

</dependencies>

<build>
    <plugins>
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-compiler-plugin</artifactId>
            <configuration>
                <source>${java-version}</source>
                <target>${java-version}</target>
            </configuration>
        </plugin>

        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-shade-plugin</artifactId>
            <version>1.7</version>
            <executions>
                <execution>
                    <phase>package</phase>
                    <goals>
                        <goal>shade</goal>
                    </goals>
                    <configuration>
                        <transformers>

                            <!-- must be SURE to do this with both spring.handlers and spring.schemas. 
                                otherwise you won't be able to use them in the spring config files. -->
                            <transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">
                                <resource>META-INF/spring.handlers</resource>
                            </transformer>

                            <transformer implementation="org.apache.maven.plugins.shade.resource.AppendingTransformer">
                                <resource>META-INF/spring.schemas</resource>
                            </transformer>

                            <transformer implementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer">
                                <mainClass>com.mycompany.test.Start</mainClass>
                            </transformer>
                        </transformers>

                        <filters>
                          <filter>
                            <artifact>bouncycastle:bcprov-jdk15</artifact>
                            <excludes>
                              <exclude>META-INF/BCKEY.DSA</exclude>
                              <exclude>META-INF/BCKEY.SF</exclude>
                              <exclude>META-INF/MANIFEST.MF</exclude>
                            </excludes>
                          </filter>
                        </filters>                          

                    </configuration>
                </execution>
            </executions>
        </plugin>

    </plugins>
</build>

jar を実行しても、他のデバッグ情報は得られません。

jarfile を検証するオプションはありますか? それともjava -verbose…のようなものを持っていますか?

再現する手順:

  1. 新しいフォルダを作成する
  2. 上記の Pom.xml をフォルダーにコピーします。
  3. 小さな `public static void main(...) を保存します。src/main/java/com/mycompany/test/Start.java に
  4. 実行するmvn clean package
  5. 実行するjava -jar target/com.mycompany.test-1.0.0-SNAPSHOT.jar

添付資料 1:

package com.mycompany.test;


public class Start
{
    public static void main(final String[] args) 
    {
        System.out.println("If you are able to get this printed with java -jar you made it. Thanks a lot! :)");
    }
}

編集1:

問題を特定するために、いくつかの依存関係を削除し始めました。しかし、私は何がうまくいかないのかを明確に理解していません。

これで、Apache POI の依存関係を一番上に移動すると問題が解決することがわかりました (pom.xml のヒント B を参照)。しかし、それでも私はその理由と何が問題なのかわかりません。

4

2 に答える 2

47

pom.xml私の側では、あなたが示した を使用してプロジェクトをビルドし、apache-poiafter を宣言するとjfreechart、あなたが言及したように、破損した JAR が得られます。これら 2 つの依存関係の順序を入れ替えると、正しい JAR が得られます。

以前に を使用した経験があり、maven-shade-pluginそれを使用したときに、作成された JAR のMETA-INFディレクトリに問題があったため、欠陥がないかどうかを確認しました。

META-INFローカル ファイル システムのどこかにディレクトリを (Total Commander を使用して) コピーしようとしましたが、ファイルのコピー中にエラーが発生したことに気付きましたMETA-INF/licences/。それらをどこかに個別にコピーしようとすると、うまくいきましたが、全体をコピーできませんでした。私の結論は、JAR/ZIP アーカイブが壊れているというものでした。

私がしたことは、その JAR を Total Commanderに入力Ctrlし ( + JAR ファイルを上書き) PgDown 名前 thirdpartylicenses.txtthirdpartylicenses.txt.wtf. これを実行すると、Total Commander はこれを保存し、JAR を再パッケージ化します (Total7zip Total Commander パッカー プラグインをインストールしました - 誰かがこれを試してもうまくいかない場合は、これをインストールして試してください)。

この後。できます。

unzip(また、Cygwin の/コマンドを使用して名前を変更せずに全体を再パックしようとしましたzipが、新しいアーカイブはまだ壊れていました。Total Commander または私が言及したプラグインはいくつかの魔法を行います。)

maven-shade-plugin破損した、または無効な ZIP/JAR アーカイブを作成するだけだと思います。理由はわかりませんが、おそらく私が説明したことが他の人にはうまくいかないかもしれませんが、私はそれについて言及したいと思ったので、私が助けることができるかもしれません.


放っておくわけにはいかないので、さらに掘り下げて答えを見つけた気がします。

不正なJAR には 65608のエントリが含まれています。適切なJAR には 65450のエントリが含まれています。

普通のZIP ファイルのエントリ数の上限はいくつだと思いますか? うん。Wiki の記事では、この制限を克服する ZIP64 形式について説明しています。

適切なJAR ではエントリが少なくなります。これは、 pom.xml. (この回答で説明されているように。

こんな感じでエントリー数を数えました。

Collections.list(new JarFile("...").entries()).size();

私は新しい ZIP64 フォーマットをサポートしているように見える Java 7 を使用していたので、誰かが Java 5 または 6 を使用して不正なJAR のエントリをカウントしようとすると、エラーが発生する可能性があります (よくわかりません)。

また、JAR を展開して実行しようとしまし。JAR全体をディレクトリに解凍し、すべてをこのように実行しました。

java -cp <dir/ com.mycompany.test.Start

魅力のように働きました。


結論。を使いすぎないでくださいmaven-shade-plugin

私はこのように自分のプロジェクトを構築する作業中のプロジェクトを持っています。

  • を使用してプロジェクトの依存関係をコピーしますmaven-dependency-plugincopy-dependenciesゴールをチェックアウトします。target/dependency私の記憶が正しければ、これにより依存関係がコピーされます。
  • プラグイン構成でこれらのオプションを使用して、これらの依存関係をエントリとして最終maven-jar-plugin的な JAR に追加します。MANIFEST.MFClass-Path

    <classpathPrefix>dependency/</classpathPrefix>
    <addClasspath>true</addClasspath>
    

    などのClass-Pathエントリがあります。dependency/<artifactId>-<version>.jar

  • この後、 を使用して、最終的な JAR とフォルダーmaven-assembly-plugin全体を含む配布 ZIP を作成します。dependency/
  • アプリケーションをデプロイすると、次のように実行できますjava -jar final.jar

主にこのソリューションを使用することを選択しました。これは、私のプロジェクトでは、ディレクトリ内に贅沢なあれこれを含む Bouncy Castle JAR を使用しているためです。META-INFを使用しmaven-shade-pluginて最終的な実行可能な JAR を作成したとき、すべての地獄が解き放たれ、メソッドが見つからず、この署名が正しくないというエラーが発生しました。

あなたもこのようなことをしているはずです。この Maven シェード ビジネスは怪しすぎて役に立ちません (しゃれが意図されています)。


これは、私が上記で説明しようとしたプロセス全体に関するブログ投稿です ( babaに感謝し ます)。将来誰かに役立つかもしれません。

于 2012-10-30T06:23:21.510 に答える
11

私はあなたのセットアップで実行しました、そしてあなたが依存関係宣言でより高いところにmvn dependency:tree移動するときそれは違いを与えるでしょう。org.apache.poi

これは、依存関係の順序に関する依存関係メカニズムの概要から抜粋したものです。

依存関係のメディエーション-これは、アーティファクトの複数のバージョンが検出されたときに使用される依存関係のバージョンを決定します。現在、Maven 2.0は「最も近い定義」の使用のみをサポートしています。これは、依存関係のツリーでプロジェクトに最も近い依存関係のバージョンを使用することを意味します。プロジェクトのPOMで明示的に宣言することにより、いつでもバージョンを保証できます。2つの依存関係バージョンが依存関係ツリーの同じ深さにある場合、Maven 2.0.8まではどちらが勝つかは定義されていませんでしたが、Maven 2.0.9以降は、宣言の順序が重要になります。最初の宣言が勝ちます。

依存関係の解決に競合があるようで、それがjarファイルの破損を引き起こしています(なぜ破損するのかわかりません)。

とにかく、ここに2つのpom間の違いがあります(左が原点、右が上にありorg.apache.poiます):

ここに画像の説明を入力してください ここに画像の説明を入力してください

写真では見づらいかもしれませんが、ズームインすると見えます。

大きな違いは、動作していないpomorg.apache.httpcomponents:httpcore:jar:4.2.1ではに依存関係がcommons-codec:commons-codec:jar:1.6あり、動作しているpomでは依存関係が。でオーバーライドされていることcommons-codec:commons-codec:jar:1.5です。

と一緒に1.6バージョンが必要なバージョンに問題があると思います。commons-codecorg.apache.poi:poi:jar:3.81.5


編集

jarファイルが破損している(jar内のエントリが多すぎる)理由を説明するこの優れた回答の後に、少なくとも特定の問題に対して機能する簡単なソリューションを追加したいと思います。

構成にタグ<minimizeJar>true</minimizeJar>を追加しmaven-shade-pluginます。

その後、java -jar target/com.mycompany.test-1.0.0-SNAPSHOT.jarコマンドラインが機能します。

于 2012-10-26T13:42:19.733 に答える