Hallo Spaß-Coder.
Im Artikel Mit Maven raus aus der Abhängigkeit haben wir einen Einstieg in die Verwaltung von Abhängigkeiten zwischen Java-Projekten gezeigt. Aber wie genau funktioniert das nun in der Praxis? Wie gehe ich vor, wenn ich Unit Tests in meinem Projekt hinzufügen möchte und dafür JUnit einbinden will? Wo finde ich diese Abhängigkeit für meine POM? Wie macht Maven das mit der Bereitstellung der Abhängigkeiten denn so genau?
Damit Maven genutzt werden kann, muss entweder die IDE-Integration aktiviert oder – so wie wir es im Laufe des Artikels beschreiben – Maven lokal installiert sein. Maven ist für verschiedene Plattformen z.B. unter apache.maven.org zu finden. Unter http://www.it-adviser.net/apache-maven-installieren/ findet ihr eine leicht verständliche Beschreibung der Installation zu Maven auf deutsch. Auf der Kommando-Zeile kann eine erfolgreiche Installation mit dem Befehl mvn -version überprüft werden.
1 2 3 4 5 6 7 8 |
>mvn -version Apache Maven 3.1.1 (893ca28a1da9d5f51ac03827af98bb730128f9f2; 2013-06-28 04:15:32+0200) Maven home: D:\Development\maven Java version: 1.7.0, vendor: Oracle Corporation Java home: D:\Programme\Java\jdk1.7.0\jre Default locale: de_DE, platform encoding: Cp1252 OS name: "windows 7", version: "6.1", arch: "amd64", family: "windows" |
Dabei muss die Version von Maven ausgegeben werden (hier: 3.1.1). Damit Maven erfolgreich betrieben werden kann, muss auch das JDK von Java installiert sein. Das habt ihr aber wahrscheinlich sowieso schon, ihr wollt ja schließlich Java programmieren 😉
Im Folgenden gehen wir davon aus, dass ihr das Grundverständnis aus dem vorangegangenen Artikel mitbringt. Falls euch was unklar ist, lest dort noch einmal nach.
Unser kleines Projekt
Beginnen wir damit, ein neues Java-Projekt anzulegen. Diesmal allerdings nicht in unserer Lieblings-IDE, sondern als Project Object Model – also POM. Wir beschreiben damit, was am Ende als Ergebnis herauskommen soll. Dies sieht im ersten Schritt wie folgt aus:
1 2 3 4 5 6 7 8 9 10 11 |
<?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/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <groupId>de.invidit</groupId> <artifactId>HelloWorld</artifactId> <version>1.0.0-SNAPSHOT</version> </project> |
Wie erinnern uns an die Beschreibung unseres Moduls als GAV, also mit der groupId, artifactId und version. Unser Projekt soll den Namen HelloWorld – was auch sonst 😉 – haben und zunächst in der Version 1.0.0 als Schnappschuss – also als Version die sich noch in der Entwicklung befindet – veröffentlicht werden.
Was passiert nun, wenn wir Maven unser Modul bauen lassen? Probieren wir es aus und rufen dazu auf der Kommando-Zeile mvn package im unserem Projektverzeichnis in dem die pom.xml liegt auf. Maven ist so freundlich und kompiliert unseren Code – in diesem Fall haben wir noch keine Programmquellen -, führt alle Unit Tests aus und erstellt die Datei HelloWorld-1.0.0-SNAPSHOT.jar im Unterordner target.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
>mvn package [INFO] Scanning for projects... [INFO] [INFO] ------------------------------------------------------------------------ [INFO] Building HelloWorld 1.0.0-SNAPSHOT [INFO] ------------------------------------------------------------------------ [INFO] [...] [INFO] ------------------------------------------------------------------------ [INFO] BUILD SUCCESS [INFO] ------------------------------------------------------------------------ [INFO] Total time: 5.803s [INFO] Finished at: Tue Sep 15 19:39:00 CEST 2015 [INFO] Final Memory: 8M/122M [INFO] ------------------------------------------------------------------------ |
Unser Projekt macht auch was
Legen wir eine typische Java-Klasse mit dem Namen HelloWorld.java und diesem Inhalt an:
1 2 3 4 5 6 7 8 9 10 11 12 |
package de.invidit.hello; public class HelloWorld { public static void main(String[] args) { System.out.println(getHelloPhrase("Maven")); } public static String getHelloPhrase(String name) { return "Hello " + name + "-World"; } } |
Die Ausgabe auf dem Standardausgabestrom soll also „Hello Maven-World“ sein. Aber wo legen wir die Datei zu unserer Klasse ab?
Maven arbeitet an vielen Stellen nach dem Prinzip „Convention over Configuration“. So auch bei den Verzeichnissen, in denen unsere Quelldateien abgelegt werden. Hier sieht Maven einen Standard für die Verzeichnisstruktur vor, an den sich alle Maven 3 Projekte halten sollten. Dieser sieht folgendermaßen aus:
- src
- main
- java
- [packages]
- [Klassen]
- [packages]
- java
- main
- test
- java
- [packages]
- [Test-Klassen]
- [packages]
- java
Unsere obige Quelldatei legen wir demnach also im Verzeichnis /src/main/java/de/invidit/ unter dem Namen HelloWorld.java an.
Was passiert nun, wenn wir mit Maven erneut unser Projekt bauen? Rufen wir mvn package also nochmal auf. Hmmm…die Ausgabe sieht genauso aus wir vorher, aber nun ist unser Code in der JAR-Datei enthalten, was wir überprüfen können, in dem wir unsere main()-Methode einfach mal aufrufen (dazu müssen wir zuvor in das target-Verzeichnis wechseln):
1 2 3 |
>java -cp HelloWorld-1.0.0-SNAPSHOT.jar de.invidit.hello.HelloWorld Hello Maven-World |
Großartig! Aber das ist nur der Anfang. Bis hierhin bietet uns Maven noch keine Vorteile gegenüber unserer IDE.
Vor dem Refaktorisieren schreiben wir natürlich Tests
Die Ausgabe auf dem Standardausgabestrom ist nicht sonderlich elegant. Wir ersetzen diese lieber durch eine Ausgabe in eine Log-Datei. Aber – bevor wir die Änderung an unserem Code vornehmen, sichern wir diese durch Unit Tests ab. Dazu nutzen wir selbstverständlich die bereits vorgestellten Bibliotheken JUnit und AsssertJ.
1 2 3 4 5 6 7 8 9 10 |
public class HelloWorldTest extends TestCase { public void testGetHelloPhraseForGivenValueReturnsCorrectPhrase() throws Exception { String expected = "Hello Maven-World"; String actual = HelloWorld.getHelloPhrase("Maven"); Assertions.assertThat(actual).isEqualTo(expected); } } |
Nach der Verzeichnisstruktur von oben liegt unsere Klasse HelloWorldTest.java im Verzeichnis /src/test/java/de/invidit/. Maven unterscheidet in der Struktur bereits zwischen Produktivem Quellcode und Testcode.
Um unser Projekt zu bauen, rufen wir erneut was auf? Genau: mvn package. Auf Grund der Standardverzeichnisstruktur von Maven wird unser Unit Test nach dem Kompilieren ausgeführt.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
> mvn package [...] [INFO] ------------------------------------------------------------- [ERROR] COMPILATION ERROR : [INFO] ------------------------------------------------------------- [ERROR] \src\test\java\de\invidit\hello\HelloWorldTest.java:[10,22] error: package junit.framework does not exist [ERROR] \src\test\java\de\invidit\hello\HelloWorldTest.java:[11,27] error: package org.assertj.core.api does not exist [ERROR] \src\test\java\de\invidit\hello\HelloWorldTest.java:[13,36] error: cannot find symbol [ERROR] \src\test\java\de\invidit\hello\HelloWorldTest.java:[20,2] error: cannot find symbol [INFO] 4 errors [INFO] ------------------------------------------------------------- [INFO] ------------------------------------------------------------------------ [INFO] BUILD FAILURE [INFO] ------------------------------------------------------------------------ [INFO] Total time: 0.998s [INFO] Finished at: Tue Sep 15 20:33:25 CEST 2015 [INFO] Final Memory: 14M/154M [INFO] ------------------------------------------------------------------------ [...] |
WTF!? Maven kommt gar nicht erst zu unserem Test, da bereits das Kompilieren abbricht. Den Fehlermeldungen können wir entnehmen, dass sowohl JUnit als auch AssertJ nicht gefunden werden. Wo bekommen wir diese jetzt her?
Suchen wir Bibliotheken, die wir mit Maven in unser Projekt einbinden wollen, können wir die Suche unter search.maven.org nutzen und dort einfach unsere Bibliothek suchen. Wer sich den Link nicht merken oder speichern möchte, setzt bei der Suche in der bevorzugten Suchmaschine „maven“ vorneweg und bekommt in der Regel einen guten Treffer. Der Vorteil bei diesen Verzeichnissen ist, dass der Eintrag für unsere POM direkt kopiert werden kann, wenn ich eine Version der Bibliothek ausgewählt habe. Nehmen wir JUnit und AssertJ als Abhängigkeit auf, sieht unsere POM so aus:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 |
<?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/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <groupId>de.invidit</groupId> <artifactId>HelloWorld</artifactId> <version>1.0.0-SNAPSHOT</version> <dependencies> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.12</version> <scope>test</scope> </dependency> <dependency> <groupId>org.assertj</groupId> <artifactId>assertj-core</artifactId> <version>2.1.0</version> <scope>test</scope> </dependency> </dependencies> </project> |
Schauen wir mal, ob Maven nun in der Lage ist, unseren Code zu kompilieren und den Test auszuführen. Erneut rufen wir mvn package auf.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 |
> mvn package [...] ------------------------------------------------------- T E S T S ------------------------------------------------------- Running de.invidit.hello.HelloWorldTest Tests run: 1, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 0.062 sec Results : Tests run: 1, Failures: 0, Errors: 0, Skipped: 0 [INFO] [INFO] --- maven-jar-plugin:2.3.1:jar (default-jar) @ HelloWorld --- [INFO] ------------------------------------------------------------------------ [INFO] BUILD SUCCESS [INFO] ------------------------------------------------------------------------ [INFO] Total time: 4.150s [INFO] Finished at: Tue Sep 15 20:50:21 CEST 2015 [INFO] Final Memory: 9M/122M [INFO] ------------------------------------------------------------------------ |
YEAH! Am Ende unseres erneuten Aufrufs erhalten wir nun die gewünschte Information, das unser Test ausgeführt wurde.
Der Test war erfolgreich;
- 1 Test wurde ausgeführt (Tests run)
- 0 Tests waren fehlerhaft (Failures und Errors)
- 0 Tests wurden übersprungen (Skipped).
Nach dem Punkt „Results :“ sehen wir noch einmal eine Zusammenfassung aller Tests aus unserem Modul. Da wir derzeit nur genau einen Test haben, unterscheiden sich die beiden Ausgaben nicht.
Wie aber hat Maven das nun gemacht? Schauen wir ein wenig weiter oben in die Ausgabe, sehen wir die Lösung dieses Rätsels:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
[INFO] ------------------------------------------------------------------------ [INFO] Building HelloWorld 1.0.0-SNAPSHOT [INFO] ------------------------------------------------------------------------ Downloading: http://repo1.maven.org/maven2/junit/junit/4.12/junit-4.12.pom Downloaded: http://repo1.maven.org/maven2/junit/junit/4.12/junit-4.12.pom (24 KB at 123.7 KB/sec) Downloading: http://repo1.maven.org/maven2/org/hamcrest/hamcrest-core/1.3/hamcrest-core-1.3.pom Downloaded: http://repo1.maven.org/maven2/org/hamcrest/hamcrest-core/1.3/hamcrest-core-1.3.pom (766 B at 11.9 KB/sec) Downloading: http://repo1.maven.org/maven2/org/hamcrest/hamcrest-parent/1.3/hamcrest-parent-1.3.pom Downloaded: http://repo1.maven.org/maven2/org/hamcrest/hamcrest-parent/1.3/hamcrest-parent-1.3.pom (2 KB at 31.1 KB/sec) Downloading: http://repo1.maven.org/maven2/org/assertj/assertj-core/2.1.0/assertj-core-2.1.0.pom Downloaded: http://repo1.maven.org/maven2/org/assertj/assertj-core/2.1.0/assertj-core-2.1.0.pom (6 KB at 63.1 KB/sec) Downloading: http://repo1.maven.org/maven2/org/assertj/assertj-parent-pom/1.3.6/assertj-parent-pom-1.3.6.pom Downloaded: http://repo1.maven.org/maven2/org/assertj/assertj-parent-pom/1.3.6/assertj-parent-pom-1.3.6.pom (15 KB at 105.7 KB/sec) Downloading: http://repo1.maven.org/maven2/junit/junit/4.12/junit-4.12.jar Downloading: http://repo1.maven.org/maven2/org/assertj/assertj-core/2.1.0/assertj-core-2.1.0.jar Downloading: http://repo1.maven.org/maven2/org/hamcrest/hamcrest-core/1.3/hamcrest-core-1.3.jar Downloaded: http://repo1.maven.org/maven2/org/hamcrest/hamcrest-core/1.3/hamcrest-core-1.3.jar (44 KB at 68.8 KB/sec) Downloaded: http://repo1.maven.org/maven2/org/assertj/assertj-core/2.1.0/assertj-core-2.1.0.jar (680 KB at 440.1 KB/sec) Downloaded: http://repo1.maven.org/maven2/junit/junit/4.12/junit-4.12.jar (308 KB at 182.6 KB/sec) [...] |
Maven lädt also die von uns definierten Abhängigkeiten ohne weiteres Zutun herunter. Woher Maven das bekommt (in diesem Fall htp://repo1.maven.org) ist wieder eine Konvention von Maven.
Wenn wir uns das Ganze ein wenig genauer anschauen, finden wir allerdings auch etwas Verwirrendes. Was ist denn hamcrest-core? Das haben wir doch gar nirgendwo hingeschrieben!?! Wo kommt das her?
Dies ist eine weitere Stärke von Maven. Die von uns verwendete Bibliothek JUnit hat selbst eine weitere Abhängigkeit, nämlich hamcrest-core. Dass das so ist, wussten wir bis eben gar nicht – und es muss uns auch nicht weiter interessieren. Maven sorgt dafür, dass wir alle weiteren Abhängigkeiten bekommen, die von den von uns genutzten Bibliotheken benötigt werden. Abhängigkeiten dieser Art werden transitive Abhängigkeiten genannt.
Eine kleine Übung zum Abschluss
Nachdem wir nun die Grundlagen gemeinsam durchgegangen sind und einen Test zur Absicherung der Funktionalität unserer Klasse geschrieben haben, gibt es zum Abschluss nun eine kleine Aufgabe.
Was müssen wir in der POM anpassen, um den Code erfolgreich packen zu können (mvn package), wenn wir den Code unserer Klasse folgendermaßen ändern:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
package de.invidit.hello; import org.apache.log4j.LogManager; import org.apache.log4j.Logger; public class HelloWorld { private static Logger logger = LogManager.getLogger(HelloWorld.class); public static void main(String[] args) { logger.info(getHelloPhrase("Maven-Dependency")); } public static String getHelloPhrase(String name) { return "Hello " + name + "-World"; } } |
Viel Spaß beim tüfteln.
Zusammenfassung und Ausblick
Wir hoffen, dass ihr durch das Selbst-Hand-Anlegen ein wenig mehr Gefühl für den Umgang mit Maven bekommen habt und sich ein bischen die Magie von Maven entfaltet hat.
In den kommenden Artikeln zur Serie werden wir noch ein paar weitere Geheimnisse lüften, etwa wohin Maven die Abhängigkeiten den herunterlädt, was dieses „package“ was wir verwendet haben zu bedeuten hat und wie wir in komplexeren Projekten unsere Abhängigkeiten effektiv verwalten können.
Habt ihre Fragen oder Anregungen, sprecht uns gerne in den Kommentaren an. Auch wenn ihr Probleme mit den Anweisungen oder der Abschlussübung haben solltet, könnt ihr gerne dort nachfragen.
Eure Spaß-Coder.
Dieser Artikel basiert neben unseren Erfahrungen auf den Ausführungen aus:
- http://maven.apache.org/
-
http://search.maven.org
- http://mvnrepository.com