Как поместить все необходимые файлы JAR в папку библиотеки внутри окончательного файла JAR с Maven?

Я использую Maven в своем автономном приложении и хочу упаковать все зависимости в моем JAR-файле в папку библиотеки, как указано в одном из ответов здесь:

Как я могу создать исполняемый JAR с зависимостями, используя Maven?

Я хочу, чтобы у моего окончательного файла JAR была папка библиотеки, которая содержит зависимости как файлы JAR, а не как maven-shade-plugin это помещает зависимости в виде папок, подобных иерархии Maven, в папку.m2.

Ну, на самом деле текущая конфигурация делает то, что я хочу, но у меня проблема с загрузкой файлов JAR при запуске приложения. Я не могу загрузить классы.

Вот моя конфигурация:

<plugins>

    <plugin>
        <groupId>org.apache.maven.plugins</groupId>
        <artifactId>maven-dependency-plugin</artifactId>
        <executions>
            <execution>
                <id>copy-dependencies</id>
                <phase>prepare-package</phase>
                <goals>
                    <goal>copy-dependencies</goal>
                </goals>
                <configuration>
                    <outputDirectory>${project.build.directory}/classes/lib</outputDirectory>
                    <overWriteReleases>false</overWriteReleases>
                    <overWriteSnapshots>false</overWriteSnapshots>
                    <overWriteIfNewer>true</overWriteIfNewer>
                </configuration>
            </execution>
        </executions>
    </plugin>

    <plugin>
        <groupId>org.apache.maven.plugins</groupId>
        <artifactId>maven-jar-plugin</artifactId>
        <configuration>
            <archive>
                <manifest>
                    <addClasspath>true</addClasspath>
                    <classpathPrefix>lib/</classpathPrefix>
                    <mainClass>com.myapp.MainClass</mainClass>
                </manifest>
            </archive>
        </configuration>
    </plugin>

    <plugin>
        <groupId>org.apache.maven.plugins</groupId>
        <artifactId>maven-compiler-plugin</artifactId>
        <configuration>
            <source>1.6</source>
            <target>1.6</target>
        </configuration>
    </plugin>

    <plugin>
        <groupId>org.apache.maven.plugins</groupId>
        <artifactId>maven-dependency-plugin</artifactId>
        <executions>
            <execution>
                <id>install</id>
                <phase>install</phase>
                <goals>
                    <goal>sources</goal>
                </goals>
            </execution>
        </executions>
    </plugin>

    <plugin>
        <groupId>org.apache.maven.plugins</groupId>
        <artifactId>maven-resources-plugin</artifactId>
        <version>2.5</version>
        <configuration>
            <encoding>UTF-8</encoding>
        </configuration>
    </plugin>

</plugins>

Проект работает нормально из Eclipse, и файлы JAR помещаются в папку библиотеки внутри моего окончательного файла JAR, как я хочу, но при запуске окончательного файла JAR из целевой папки я всегда получаю ClassNotFoundException:

Exception in thread "main" java.lang.NoClassDefFoundError: org/springframework/context/ApplicationContext
Caused by: java.lang.ClassNotFoundException: org.springframework.context.ApplicationContext
        at java.net.URLClassLoader$1.run(Unknown Source)
        at java.security.AccessController.doPrivileged(Native Method)
        at java.net.URLClassLoader.findClass(Unknown Source)
        at java.lang.ClassLoader.loadClass(Unknown Source)
        at sun.misc.Launcher$AppClassLoader.loadClass(Unknown Source)
        at java.lang.ClassLoader.loadClass(Unknown Source)
Could not find the main class: com.myapp.MainClass. Program will exit.

Как я могу исправить это исключение?

8 ответов

Решение

По этой ссылке:

Как: Eclipse Maven установить сборку JAR с зависимостями

я обнаружил, что это нереализуемое решение, потому что загрузчик классов не загружает jar-файлы из jar-файлов, поэтому я думаю, что распакую зависимости внутри jar-файла.

Следующее моё решение. Проверьте это, если это работает для вас:

<plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-dependency-plugin</artifactId>
    <executions>
        <execution>
            <id>copy-dependencies</id>
            <phase>prepare-package</phase>
            <goals>
                <goal>copy-dependencies</goal>
            </goals>
            <configuration>
                <outputDirectory>${project.build.directory}/classes/lib</outputDirectory>
                <overWriteReleases>false</overWriteReleases>
                <overWriteSnapshots>false</overWriteSnapshots>
                <overWriteIfNewer>true</overWriteIfNewer>
            </configuration>
        </execution>
    </executions>
</plugin>

<plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-jar-plugin</artifactId>
    <configuration>
        <archive>
            <manifest>
                <addClasspath>true</addClasspath>
                <!-- <classpathPrefix>lib</classpathPrefix> -->
                <!-- <mainClass>test.org.Cliente</mainClass> -->
            </manifest>
            <manifestEntries>
                <Class-Path>lib/</Class-Path>
            </manifestEntries>
        </archive>
    </configuration>
</plugin>

Первый плагин помещает все зависимости в папку target/classes/lib, а второй включает папку библиотеки в окончательный файл JAR и настраивает Manifest.mf файл.

Но тогда вам нужно будет добавить собственный код загрузки классов для загрузки файлов JAR.

Или, чтобы избежать пользовательской загрузки классов, вы можете использовать "${project.build.directory}/lib, но в этом случае у вас нет зависимостей внутри конечного файла JAR, что противоречит цели.

Прошло два года с тех пор, как был задан вопрос. Проблема вложенных JAR-файлов все же сохраняется. Надеюсь, это кому-нибудь поможет.

Обновлено:

<build> 
  <plugins> 
    <plugin> 
    <artifactId>maven-dependency-plugin</artifactId> 
    <executions> 
      <execution> 
        <phase>install</phase> 
          <goals> 
            <goal>copy-dependencies</goal> 
          </goals> 
          <configuration> 
             <outputDirectory>${project.build.directory}/lib</outputDirectory> 
          </configuration> 
        </execution> 
      </executions> 
    </plugin> 
  </plugins> 
</build> 

Самый простой и эффективный способ - использовать Uber-плагин, например так:

          <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-shade-plugin</artifactId>
            <executions>
                <execution>
                    <phase>package</phase>
                    <goals>
                        <goal>shade</goal>
                    </goals>
                </execution>
            </executions>
            <configuration>
                <finalName>uber-${artifactId}-${version}</finalName>
            </configuration>
        </plugin>

Вы будете де-нормализовать все в одном файле JAR.

Плагин исполняемого упаковщика maven может использоваться именно для этой цели: создания автономных java-приложений, содержащих все зависимости в виде JAR-файлов в определенной папке.

Просто добавьте следующее к вашему pom.xml внутри <build><plugins> раздел (не забудьте заменить значение mainClass соответственно):

<plugin>
    <groupId>de.ntcomputer</groupId>
    <artifactId>executable-packer-maven-plugin</artifactId>
    <version>1.0.1</version>
    <configuration>
        <mainClass>com.example.MyMainClass</mainClass>
    </configuration>
    <executions>
        <execution>
            <goals>
                <goal>pack-executable-jar</goal>
            </goals>
        </execution>
    </executions>
</plugin>

Встроенный файл JAR находится по адресу target/<YourProjectAndVersion>-pkg.jar после запуска mvn package, Все его зависимости времени компиляции и времени выполнения будут включены в lib/ папка внутри файла JAR.

Отказ от ответственности: я автор плагина.

Вот как я это делаю:

    <plugin>
            <artifactId>maven-assembly-plugin</artifactId>
            <version>2.2</version>
            <configuration>
                <appendAssemblyId>false</appendAssemblyId>
                <descriptorRefs>
                    <descriptorRef>jar-with-dependencies</descriptorRef>
                </descriptorRefs>
                <archive>
                    <manifest>
                        <mainClass>com.project.MainClass</mainClass>
                    </manifest>
                </archive>
            </configuration>
        </plugin>

И тогда я просто бегу:

mvn assembly:assembly

Я нашел этот ответ на вопрос:

http://padcom13.blogspot.co.uk/2011/10/creating-standalone-applications-with.html

Вы не только получаете зависимые файлы lib в папке lib, вы также получаете директорию bin с исполняемым файлом unix и dos.

В конечном итоге исполняемый файл вызывает java с аргументом -cp, в котором также перечислены все ваши зависимые библиотеки.

Вся партия находится в папке appasembly внутри целевой папки. Эпическая.

============= Да, я знаю, что это старая ветка, но она по-прежнему высоко ценится в результатах поиска, поэтому я подумала, что она может помочь кому-то вроде меня.

Это явно проблема classpath. Примите во внимание, что путь к классу должен немного измениться, когда вы запускаете свою программу вне IDE. Это связано с тем, что среда IDE загружает другие JAR-файлы относительно корневой папки вашего проекта, в то время как в случае окончательного JAR-файла это обычно не так.

В этих ситуациях мне нравится делать JAR вручную. У меня уходит не более 5 минут, и это всегда решает проблему. Я не предлагаю вам сделать это. Найти способ использовать Maven, это его цель.

Другие вопросы по тегам