Maven неправильно устанавливает classpath для зависимостей - PullRequest
18 голосов
/ 14 января 2011

Имя ОС: "linux" версия: "2.6.32-27-generic" arch: "i386" Семейство: "unix"

Apache Maven 2.2.1 (r801777; 2009-08-06 12: 16: 01-0700)

Java-версия: 1.6.0_20

Я пытаюсь использовать зависимость mysql с maven в Ubuntu. Если я переместу файл «mysql-connector-java-5.1.14.jar», загруженный maven, в мою папку $ JAVA_HOME / jre / lib / ext /, все будет хорошо, когда я запусту jar.

Я думаю, что я должен иметь возможность просто указать зависимость в файле pom.xml, а maven должен позаботиться об автоматической установке classpath для jar-зависимостей. Это неправильно?

Мой файл pom.xml выглядит так:

<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/xsd/maven-4.0.0.xsd">
  <modelVersion>4.0.0</modelVersion>

  <groupId>com.ion.common</groupId>
  <artifactId>TestPreparation</artifactId>
  <version>1.0-SNAPSHOT</version>
  <packaging>jar</packaging>

  <name>TestPrep</name>
  <url>http://maven.apache.org</url>

  <properties>
    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
  </properties>

  <build>
    <plugins>
      <plugin>
        <groupId>org.apache.maven.plugins</groupId>
        <artifactId>maven-jar-plugin</artifactId>

        <configuration>
          <archive>
            <manifest>
              <addClasspath>true</addClasspath>
              <mainClass>com.ion.common.App</mainClass>
            </manifest>
          </archive>
        </configuration>

      </plugin>
    </plugins>
  </build>

  <dependencies>

    <!-- JUnit testing dependency -->
    <dependency>
      <groupId>junit</groupId>
      <artifactId>junit</artifactId>
      <version>3.8.1</version>
      <scope>test</scope>
    </dependency>

    <!-- MySQL database driver -->
    <dependency>
      <groupId>mysql</groupId>
      <artifactId>mysql-connector-java</artifactId>
      <version>5.1.14</version>
      <scope>compile</scope>
    </dependency>

  </dependencies>
</project>

Команда "mvn package" создает его без проблем, и я могу его запустить, но когда приложение пытается получить доступ к базе данных, появляется эта ошибка:

java.lang.ClassNotFoundException: com.mysql.jdbc.Driver
        at java.net.URLClassLoader$1.run(URLClassLoader.java:217)
        at java.security.AccessController.doPrivileged(Native Method)
        at java.net.URLClassLoader.findClass(URLClassLoader.java:205)
        at java.lang.ClassLoader.loadClass(ClassLoader.java:321)
        at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:294)
        at java.lang.ClassLoader.loadClass(ClassLoader.java:266)
        at java.lang.Class.forName0(Native Method)
        at java.lang.Class.forName(Class.java:186)
        at com.ion.common.Functions.databases(Functions.java:107)
        at com.ion.common.App.main(App.java:31)

Строка, на которой происходит сбой:

Class.forName("com.mysql.jdbc.Driver");

Может кто-нибудь сказать мне, что я делаю неправильно или как это исправить?

Ответы [ 4 ]

19 голосов
/ 14 января 2011

Рагурам толкнул меня в правильном направлении. Чтобы заставить maven позаботиться о автоматическом копировании файлов jar, добавьте этот код в тег в файле pom.xml:

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

Подробнее об этом можно узнать здесь: https://maven.apache.org/plugins/maven-dependency-plugin/usage.html

Было бы неплохо получить maven для упаковки банок, но этого достаточно, чтобы ответить на этот вопрос. Связанные ответы на stackoverflow:

Сборка исполняемого кувшина с мавеном?

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

13 голосов
/ 02 февраля 2013

Я знаю, что этот вопрос старый, но он появляется в верхней части поисков, чтобы заставить Maven правильно установить зависимости с версиями -SNAPSHOT, и мне пришлось уточнить принятое решение, чтобы заставить мое разрешение пути к классам работать правильно.

Проблема, с которой я столкнулся, заключалась в том, что maven-jar-plugin включал resolvedVersion зависимости (например, -. Jar), ​​а maven-dependency-plugin (начиная с версии 2.5.1) копирует зависимости, сохраняя их baseVersion - -SNAPSHOT.jar). (См. https://jira.codehaus.org/browse/MDEP-380 для получения дополнительной информации об этом улучшении.)

Чтобы все заработало, мне пришлось отключить это поведение следующим образом:

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

        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-dependency-plugin</artifactId>
            <executions>
                <execution>
                    <id>copy-dependencies</id>
                    <phase>package</phase>
                    <goals>
                        <goal>copy-dependencies</goal>
                    </goals>
                    <configuration>
                        <useBaseVersion>false</useBaseVersion>
                    </configuration>
                </execution>
            </executions>
        </plugin>
    </plugins>
    ...

Эта конфигурация привела к тому, что зависимости были скопированы в ${project.build.directory}/dependency с их resolvedVersion, совпадающими с blasspath, установленным в META-INF/MANIFEST.MF с помощью maven-jar-plugin. Надеюсь, это поможет кому-то в будущем.

2 голосов
/ 14 января 2011

Maven правильно устанавливает classpath для зависимостей правильно, но без префикса с расположением хранилища. Это будет выглядеть в вашем файле Manifest.

Class-Path: mysql-connector-java-5.1.14.jar

Вы должны поместить зависимые банки в ту же папку, что и запущенный файл.

См. примеры

0 голосов
/ 23 мая 2019

Или, если вы используете maven-shade-plugin, обходной путь - добавить фильтр для включения отсутствующих файлов:

        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-shade-plugin</artifactId>
            <version>3.1.0</version>
            <executions>
                <execution>
                    ...
                    <configuration>
                        ...
                        <filters>
                            ...                                
                            <filter>
                                <artifact>com.sun.istack</artifact>
                                <includes>
                                    <include>**</include>
                                </includes>
                            </filter>
...