Rheinwerk Computing < openbook >


 
Inhaltsverzeichnis
Materialien zum Buch
Vorwort
1 Java ist auch eine Sprache
2 Imperative Sprachkonzepte
3 Klassen und Objekte
4 Arrays und ihre Anwendungen
5 Der Umgang mit Zeichen und Zeichenketten
6 Eigene Klassen schreiben
7 Objektorientierte Beziehungsfragen
8 Schnittstellen, Aufzählungen, versiegelte Klassen, Records
9 Ausnahmen müssen sein
10 Geschachtelte Typen
11 Besondere Typen der Java SE
12 Generics<T>
13 Lambda-Ausdrücke und funktionale Programmierung
14 Architektur, Design und angewandte Objektorientierung
15 Java Platform Module System
16 Die Klassenbibliothek
17 Einführung in die nebenläufige Programmierung
18 Einführung in Datenstrukturen und Algorithmen
19 Einführung in grafische Oberflächen
20 Einführung in Dateien und Datenströme
21 Einführung ins Datenbankmanagement mit JDBC
22 Bits und Bytes, Mathematisches und Geld
23 Testen mit JUnit
24 Die Werkzeuge des JDK
A Java SE-Module und Paketübersicht
Stichwortverzeichnis


Buch bestellen
Ihre Meinung?



Spacer
<< zurück
Java ist auch eine Insel von Christian Ullenboom

Einführung, Ausbildung, Praxis
Buch: Java ist auch eine Insel


Java ist auch eine Insel

Pfeil 23 Testen mit JUnit
Pfeil 23.1 Softwaretests
Pfeil 23.1.1 Vorgehen beim Schreiben von Testfällen
Pfeil 23.2 Das Test-Framework JUnit
Pfeil 23.2.1 Test-Driven Development und Test-First
Pfeil 23.2.2 Testen, implementieren, testen, implementieren, testen, freuen
Pfeil 23.2.3 JUnit-Tests ausführen
Pfeil 23.2.4 assert*(…)-Methoden der Klasse Assertions
Pfeil 23.2.5 Exceptions testen
Pfeil 23.2.6 Grenzen für Ausführungszeiten festlegen
Pfeil 23.2.7 Beschriftungen mit @DisplayName
Pfeil 23.2.8 Verschachtelte Tests
Pfeil 23.2.9 Tests ignorieren
Pfeil 23.2.10 Mit Methoden der Assumptions-Klasse Tests abbrechen
Pfeil 23.2.11 Parametrisierte Tests
Pfeil 23.3 Java-Assertions-Bibliotheken und AssertJ
Pfeil 23.3.1 AssertJ
Pfeil 23.4 Aufbau größerer Testfälle
Pfeil 23.4.1 Fixtures
Pfeil 23.4.2 Sammlungen von Testklassen und Klassenorganisation
Pfeil 23.5 Wie gutes Design das Testen ermöglicht
Pfeil 23.6 Dummy, Fake, Stub und Mock
Pfeil 23.7 JUnit-Erweiterungen, Testzusätze
Pfeil 23.8 Zum Weiterlesen
 

Zum Seitenanfang

23.2    Das Test-Framework JUnit Zur vorigen ÜberschriftZur nächsten Überschrift

Oracle definiert kein allgemeines Standard-Framework zur Definition von Unit-Testfällen, und es bietet auch keine Ablaufumgebung für Testfälle. Diese Lücke füllen Test-Frameworks, wobei das populärste im Java-Bereich das freie quelloffene JUnit (https://junit.org/junit5/) ist. Mehr als 60 % aller quelloffenen Projekte unter GitHub referenzieren diese Bibliothek.

JUnit-Versionen

Kent Beck und Erich Gamma begannen im Jahr 2000 mit der Entwicklung des JUnit-Frameworks. Die aktuellen Änderungen kommen von diversen Entwicklern. Der ursprüngliche Zweig JUnit 3 nutzt keine Annotationen. Das änderte sich 2006 mit der Version JUnit 4, die aktuell die größte Nutzerbasis hat. Da JUnit 4 eine monolithische Bibliothek ist, wurde JUnit 5 geschrieben und eine Modularisierung vorgenommen: JUnit 5 besteht aus den Teilen JUnit Platform, JUnit Jupiter und JUnit Vintage. JUnit 5 benötigt mindestens Java 8 und benennt auch Annotationen um, sodass es einen gewissen Migrationsaufwand bei der Umstellung von JUnit 4 gibt.

JUnit aufnehmen

Wir wollen mit der aktuellen Version JUnit 5 arbeiten. Da JUnit kein Teil der Java SE ist, müssen wir Bibliotheken mit in den Klassenpfad aufnehmen. Wir können natürlich die JAR-Dateien von Hand herunterladen und in den Klassenpfad aufnehmen, doch über Maven und die IDE geht es einfacher und schneller. Wir fügen folgende Abhängigkeit in die POM-Datei ein:

Listing 23.1     pom.xml, Ergänzung

<dependency>

<groupId>org.junit.jupiter</groupId>

<artifactId>junit-jupiter-engine</artifactId>

<version>5.8.1</version>

<scope>test</scope>

</dependency>

Die Standard-IDEs Eclipse und IntelliJ bringen JUnit gleich mit und bieten Wizards an, mit denen sich einfach Testfälle aus vorhandenen Klassen erstellen lassen. Per Tastendruck lassen sich Testfälle abarbeiten, und ein farbiger Balken zeigt direkt an, ob wir unsere Arbeit gut gemacht haben.

 

Zum Seitenanfang

23.2.1    Test-Driven Development und Test-First Zur vorigen ÜberschriftZur nächsten Überschrift

Unser JUnit-Beispiel wollen wir nach einem ganz speziellen Ansatz entwickeln, der sich Test-First nennt. Dabei wird der Testfall noch vor der eigentlichen Implementierung geschrieben. Die Reihenfolge mit dem Test-First-Ansatz sieht (etwas erweitert) so aus:

  1. Überlege, welche Klasse und Methode geschrieben werden soll. Lege Quellcode für die Klasse und für die Variablen, Methoden bzw. Konstruktoren an, sodass sich die Compilationseinheit übersetzen lässt. Die Codeblöcke sind leer, enthalten aber mitunter eine return-Anweisung mit Rückgabe, sodass die Typen und Methoden bzw. Konstruktoren »da« sind, aber keine Funktionalität besitzen.

  2. Schreibe die API-Dokumentation, und dokumentiere, welche Funktion und Bedeutung Parameter, Rückgaben und Ausnahmen haben.

  3. Teste die API an einem Beispiel, das zeigt, ob sich die Klasse mit Eigenschaften »natürlich« anfühlt. Falls nötig, wechsle zu Punkt 1, und passe die Eigenschaften an.

  4. Implementiere eine Testklasse.

  5. Implementiere die Logik des eigentlichen Programms.

  6. Gibt es durch die Implementierung neue Dinge, die ein Testfall testen sollte? Wenn ja, erweitere den Testfall.

  7. Führe die Tests aus, und wiederhole ab Schritt 5, bis alles fehlerfrei läuft.

Der Test-First-Ansatz hat den großen Vorteil, dass er überschnelle Entwickler, die, ohne groß zu denken, zur Tastatur greifen, dann implementieren und nach 20 Minuten wieder alles ändern, zum Nachdenken zwingt. Große Änderungen kosten Zeit und somit Geld, und Test-First verringert die Notwendigkeit späterer Änderungen. Denn wenn Entwickler Zeit in die API-Dokumentation investieren und Testfälle schreiben, dann haben sie eine sehr gute Vorstellung von der Arbeitsweise der Klasse, und große Änderungen sind seltener.

Der Test-First-Ansatz ist eine Anwendung von Test-Driven Development (TDD). Hier geht es darum, die Testbarkeit gleich als Ziel bei der Softwareentwicklung zu definieren. Hieran krankten frühere Entwicklungsmodelle, etwa das wohlbekannte Wasserfallmodell, das das Testen an das Ende – nach Analyse, Design und Implementierung – stellte. Die Konsequenz dieser Reihenfolge war oft ein großer Klumpen Programmcode, der unmöglich zu testen war. Mit TDD soll das nicht mehr passieren. Heutzutage sollten sich Entwickler bei jeder Architektur, jedem Design und jeder Klasse gleich zu Beginn überlegen, wie das Ergebnis zu testen ist. Untersuchungen zeigen: Mit TDD ist das Design signifikant besser.

Zu der Frage, wann Tests durchgeführt werden sollen, lässt sich nur eines festhalten: so oft wie möglich. Denn je eher ein Test durch eine falsche Programmänderung fehlschlägt, desto eher kann der Fehler behoben werden. Gute Zeitpunkte sind daher vor und hinter größeren Designänderungen und auf jeden Fall vor dem Einpflegen in die Versionsverwaltung. Im modernen Entwicklungsprozess gibt es einen Rechner, auf dem eine Software zur kontinuierlichen Integration läuft (engl. continuous integration). Diese Systeme integrieren einen Build-Server, der die Quellen automatisch aus einer Versionsverwaltung auscheckt, compiliert und dann Testfälle und weitere Metriken laufen lässt. Diese Software übernimmt dann einen Integrationstest, da hier alle Module der Software zu einer Gesamtheit zusammengebaut werden und so Probleme aufgezeigt werden, die sich vielleicht bei isolierten Tests auf den Entwicklermaschinen nicht zeigen.

 

Zum Seitenanfang

23.2.2    Testen, implementieren, testen, implementieren, testen, freuen Zur vorigen ÜberschriftZur nächsten Überschrift

Bisher bietet Java keine einfache Funktion, die Strings umdreht. Unser erstes JUnit-Beispiel soll daher um eine neue Klasse Strings mit einer statische Methode reverse(String) gestrickt werden.

Zu testende Klasse schreiben

Nach dem TDD-Ansatz implementieren wir eine Klasse und die Methode, sodass korrekt übersetzt werden kann, aber alles zunächst ohne Funktionalität ist. (Auf die komplette API-Dokumentation verzichtet das Beispiel.)

Listing 23.2     src/main/java/com/tutego/insel/junit/util/Strings.java, Strings

public class Strings {



/**

* Reverses a given String.

*/


public static String reverse( String string ) {

return null;

}

}

Der Name und der Parametertyp »fühlen« sich richtig an, und gegen diese eigene API lässt sich nun der Testfall schreiben.

JUnit-Testfall schreiben

Spontan fällt uns ein, dass ein Leer-String umgedreht natürlich auch einen Leer-String ergibt und die Zeichenkette »abc« daher umgedreht »cba« ergibt. Unser Ziel ist es, eine möglichst gute Abdeckung aller Fälle zu bekommen. Wenn wir Fallunterscheidungen im Programmcode vermuten, sollten wir versuchen, so viele Testfälle zu finden, dass alle diese Fallunterscheidungen abgelaufen werden. Interessant sind beim Eingeben immer Sonderfälle bzw. Grenzen von Wertebereichen. (Unsere Methode gibt da nicht viel her, aber wenn wir etwa eine Substring-Funktion haben, lassen sich schnell viele Methodenübergaben finden, die interessant sind.)

Listing 23.3     src/test/java/com/tutego/insel/junit/util/StringsTest.java

package com.tutego.insel.junit.util;



import org.junit.jupiter.api.Test;

import static org.junit.jupiter.api.Assertions.*;



class StringsTest {



@Test

void reverse_non_null_string() {

// given

String emptyString = "";



// when

String reversed = Strings.reverse( emptyString );



// then

assertEquals( "", reversed );



assertEquals( "cba", Strings.reverse( "abc" ) );

}

}

Die Klasse zeigt fünf Besonderheiten:

  1. Die Testklasse endet mit dem Suffix Test, aber das ist nur Konvention und nicht zwingend. Ab JUnit 5 muss die Klasse nicht mehr zwingend öffentlich sein.

  2. Die Methoden, die sich einzelne Szenarien vornehmen und die Klassen bzw. Methoden testen, tragen die Annotation @Test. Die Annotation hat in JUnit 5 keine Attribute mehr, unter JUnit 4 konnten noch eine maximale Ausführungszeit und erwartete Ausnahmen festgesetzt werden. Ab JUnit 5 müssen die Methoden nicht mehr öffentlich sein.

  3. Unterschiedliche Autoren verwenden unterschiedliche Benennungen der Testmethoden, eine zwingende Namenskonvention gibt es nicht. Hier ist eine Notation gewählt, bei der das Präfix die zu testende Methode benennt, gefolgt vom dem, was der Test prüft, nämlich nichtleere Strings umzudrehen. Um die Lesbarkeit zu verbessern, trennt ein Unterstrich die Segmente. Das führt zum Methodennamen reverse_non_null_string(). Eine traditionelle Namenskonvention ist, dass die Methode, die den Test enthält, mit dem Präfix test beginnt und mit dem Namen der Methode endet, die sie testet. Nach diesem Bauplan könnte unsere Methode testReverse() heißen.

  4. JUnit bietet eine Reihe von assert*(…)-Methoden, die den erwarteten Zustand mit dem Ist-Zustand vergleichen. Gibt es Abweichungen, folgt eine Ausnahme. assertEquals(…) nimmt intern einen equals(…)-Vergleich der beiden Objekte vor. Wenn demnach Strings.reverse("") die leere Zeichenkette "" liefert, ist alles in Ordnung, und der Test wird fortgesetzt.

  5. Der statische Import aller statischen Eigenschaften der Klasse org.junit.jupiter.api. Assertions kürzt die Schreibweise ab, sodass im Programm statt Assertions.assertEquals(…) nur assertEquals(…) geschrieben werden kann.

 

Zum Seitenanfang

23.2.3    JUnit-Tests ausführen Zur vorigen ÜberschriftZur nächsten Überschrift

Zum Ausführen von Testfällen gibt es unterschiedliche Möglichkeiten:

  1. In einer Entwicklungsumgebung lassen sich die Tests leicht ausführen. Eclipse oder IntelliJ zeigt zum Beispiel die Ergebnisse in einer JUnit-View an und bieten mit einem grünen oder roten Balken direktes visuelles Feedback.

  2. JUnit kann über den Console Launcher die Tests von der Kommandozeile ausführen.[ 280 ](Details siehe http://junit.org/junit5/docs/current/user-guide/#running-tests-console-launcher)

  3. Eine Testausführung in der IDE ist in der Entwicklung praktisch. Doch in einer professionellen Build-Infrastruktur werden die Tests mit Maven oder Gradle angestoßen. Bei Maven ist die Testausführung Teil einer Phase im Lebenszyklus.

Eigene Implementierung überarbeiten

In unserem Beispiel wird der Testlauf fehlschlagen, weil wir keine funktionierende Implementierung der reverse(…)-Methode haben. Bisher steht im Rumpf:

public static String reverse( String string ) {

return null;

}

Ändern wir das:

public static String reverse( String string ) {

return new StringBuilder( string ).reverse().toString();

}

Beim nächsten Testlauf gibt es keinen Fehler mehr.

 

Zum Seitenanfang

23.2.4    assert*(…)-Methoden der Klasse Assertions Zur vorigen ÜberschriftZur nächsten Überschrift

Assertions ist die Klasse mit diversen assert*(…)-Methoden, die immer dann einen AssertionFailedError auslösen, wenn ein aktueller Wert nicht so wie der gewünschte war. Der JUnit-Runner fängt alle AssertionFailedErrors ab und speichert sie für die Statistik. Bis auf drei Ausnahmen beginnen alle Methoden der Klasse Assertions mit dem Präfix assert – zwei andere heißen fail(…), und eine heißt isArray(…). Die assert*(…)-Methoden gibt es einmal mit einer Testmeldung, die dann erscheint, wenn JUnit extra eine Meldung angeben soll, und einmal ohne, wenn keine Extrameldung gefragt ist. Die Meldungen können auch über einen Supplier<String> geliefert werden, das ist in der folgenden Dokumentation aber nicht mit aufgeführt.

Ist etwas wahr oder falsch?

Eigentlich reicht zum Testen die Methode assertTrue(boolean condition) aus. Ist die Bedingung wahr, so ist alles in Ordnung. Wenn nicht, gibt es intern einen AssertionFailedError, eine Unterklasse von java.lang.AssertionError, die selbst Unterklasse von java.lang.Error ist.

class org.junit.jupiter.api.Assertions
  • static void assertTrue(boolean condition)

  • static void assertTrue(String message, boolean condition)

  • static void assertFalse(boolean condition)

  • static void assertFalse(String message, boolean condition)

  • Außerdem gibt es assert[True|False](…) mit einem BooleanSupplier.

Ist etwas null?

Um es Entwicklern etwas komfortabler zu machen, bietet JUnit sechs Kategorien von Hilfsmethoden. Zunächst sind es assertNull(…) und assertNotNull(…), die testen, ob das Argument null bzw. nicht null ist. Ein Aufruf von assertNull(Object object) ist dann nichts anderes als assertTrue(object == null):

  • static void assertNotNull(Object object)

  • static void assertNotNull(String message, Object object)

  • static void assertNull(Object object)

  • static void assertNull(String message, Object object)

Sind Objekte identisch?

Die nächste Kategorie testet, ob das Objekt identisch mit einem anderen Objekt ist – es geht hier nicht um equals(…)-gleich:

  • static void assertNotSame(Object unexpected, Object actual)

  • static void assertNotSame(String message, Object unexpected, Object actual)

  • static void assertSame(Object expected, Object actual)

  • static void assertSame(String message, Object expected, Object actual)

Sind Objekte gleichwertig?

Statt eines Referenztests führen die folgenden Methoden einen equals(…)-Vergleich durch:

  • static void assertEquals(Object expected, Object actual)

  • static void assertEquals(String message, Object expected, Object actual)

Sind primitive Werte gleich?

Zum Testen von primitiven Datentypen gibt es im Grunde nur drei Methoden: einmal für den Datentyp long (alles »Kleine« wird automatisch typangepasst) und float sowie double:

  • static void assertEquals(long expected, long actual)

  • static void assertEquals(float|double expected, float|double actual,

    float|double delta)

  • static void assertEquals(String message, long expected, long actual)

  • static void assertEquals(String message, float|double expected,

    float|double actual, float|double delta)

Bei dem Vergleich von Fließkommazahlen muss bei assertEquals(…) ein Delta-Wert mitgegeben werden, in dem sich das Ergebnis bewegen muss. Das trägt der Tatsache Rechnung, dass vielleicht in der Bildschirmausgabe zwei Zahlen gleich aussehen, jedoch nicht bitweise gleich sind, wenn sich etwa kleine Rechenfehler akkumuliert haben. Sind jedoch die Fließkommazahlen in einem Wrapper, also etwa Double, verpackt, leitet ja assertEquals(…) den Test nur an die equals(…)-Methode der Wrapper-Klasse weiter, die natürlich kein Delta berücksichtigt.

Sind Arrays gleich?

Weitere Methoden vergleichen Array-Inhalte; BCSIL steht stellvertretend für byte, char, short, int, long:

  • static void assertArrayEquals(BCSIL[] expecteds, byte[] actuals)

  • static void assertArrayEquals(String message, byte[] expecteds, BCSIL[] actuals)

  • static void assertArrayEquals(String message, long[] expecteds, long[] actuals)

  • static void assertArrayEquals(String message, Object[] expecteds, Object[] actuals)

Neben den assertEquals(…)-Methoden gibt es für einige Varianten Negationen:

  • static void assertNotEquals(long unexpected, long actual)

  • static void assertNotEquals(float unexpected, float actual, float delta)

  • static void assertNotEquals(double unexpected, double actual, double delta)

  • static void assertNotEquals(Object unexpected, Object actual)

  • static void assertNotEquals(String message, long unexpected, long actual)

  • static void assertNotEquals(String message, float unexpected, float actual, float delta)

  • static void assertNotEquals(String message, double unexpected, double actual,

    double delta)

  • static void assertNotEquals(String message, Object unexpected, Object actual)

Ist das alles wahr oder falsch?

In JUnit 5 ist ein neuer Typ org.junit.jupiter.api.function.Executable eingezogen, mit dem sich ein beliebiger Block Code ausdrücken lässt. Assertions nimmt diesen Typ an über:

  • static void assertAll(Executable… executables)

  • static void assertAll(Stream<Executable> executables)

  • static void assertAll(String heading, Executable… executables)

  • static void assertAll(String heading, Stream<Executable> executables)

Ein Executable ist eine funktionale Schnittstelle mit einer Methode void execute() throws Throwable. Das Praktische bei assertAll(…) ist, dass sie alle Blöcke ausführt, selbst wenn es bei einem einen Fehler gibt – normalerweise bricht eine @Test-Methode dann ab. Eine Ausnahme beendet nur das aktuelle execute()Assertions.assertAll(…) fängt diese Ausnahme auf, meldet den Fehler bei der Abarbeitung des Testcodes, macht aber sonst weiter.

 

Zum Seitenanfang

23.2.5    Exceptions testen Zur vorigen ÜberschriftZur nächsten Überschrift

Während der Implementierung fallen oft Dinge auf, die die eigentliche Implementierung noch nicht berücksichtigt. Dann sollte sofort diese neu gewonnene Erkenntnis in den Testfall einfließen. In unserem Beispiel soll das bedeuten, dass bisher nicht wirklich geklärt ist, was bei einem null-Argument passieren soll. Bisher gibt es eine NullPointerException, und das ist auch völlig in Ordnung, aber in einem Testfall steht das bisher nicht, dass auch wirklich eine NullPointerException folgt. Diese Fragestellung legt den Fokus auf eine gern vergessene Seite des Testens, denn Testautoren dürfen sich nicht nur darauf konzentrieren, was die Implementierung denn so alles richtig machen soll – der Test muss auch kontrollieren, ob im Fehlerfall auch dieser korrekt gemeldet wird. Wenn es nicht in der Spezifikation steht, dürfen auf keinen Fall falsche Werte geradegebügelt werden: Falsche Werte müssen immer zu einer Ausnahme oder zu einem wohldefinierten Verhalten führen.

Wir wollen unser Beispiel so erweitern, dass reverse(null) eine IllegalArgumentException auslöst. Auf zwei Arten lässt sich testen, ob die erwartete IllegalArgumentException auch wirklich kommt.

Versuch und fail(…)

Die erste Variante:

Listing 23.4     com/tutego/insel/junit/utils/StringsTest.java, reverse_null_string_1()

@Test void reverse_null_string_1() {

try {

Strings.reverse( null );

fail( "reverse(null) should throw IllegalArgumentException" );

}

catch ( IllegalArgumentException e ) { /* Ignore */ }

}

Führt reverse(null) zur Ausnahme, was ja gewollt ist, dann wird der catch-Block die IllegalArgumentException einfach auffangen und ignorieren, und dann geht es in der Testfunktion mit anderen Dingen weiter. Sollte keine Ausnahme folgen, so wird die Anweisung nach dem reverse(…)-Aufruf ausgeführt, und die ist fail(…). Diese Methode löst eine JUnit-Ausnahme mit einer Meldung aus und signalisiert dadurch, dass im Test etwas nicht stimmte.

assertThrows(…)

Eine zweite Möglichkeit bietet JUnit mit assertThrows(…):

Listing 23.5     com/tutego/insel/junit/util/StringsTest.java, reverse_null_string_2()

@Test void reverse_null_string_2() {

assertThrows( IllegalArgumentException.class, () -> {

Strings.reverse( null );

} );

}

Insgesamt gibt es in der Klasse Assertions die folgenden Methoden zum Prüfen von Ausnahmen:

  • static <T extends Throwable> T assertThrows(Class<T> expectedType,

    Executable executable)

  • static <T extends Throwable> T assertThrows(Class<T> expectedType,

    Executable executable, String message)

  • static <T extends Throwable> T assertThrows(Class<T> expectedType,

    Executable executable, Supplier<String> messageSupplier)

 

Zum Seitenanfang

23.2.6    Grenzen für Ausführungszeiten festlegen Zur vorigen ÜberschriftZur nächsten Überschrift

Nach großen Refactorings kann die Software funktional ihre Tests bestehen, ist aber vielleicht viel langsamer geworden. Dann stellt sich die Frage, ob es im Sinne des Anforderungskatalogs noch korrekt ist, wenn ein performantes Programm nach einer Änderung wie eine Schnecke läuft.

Um Laufzeitveränderungen als Gültigkeitskriterium einzuführen, kann der Test in ein assertTimeout(…) bzw. assertTimeoutPreemptively(…) gesetzt werden. Beide Methoden erwarten nach einer gegebenen Duration ein Executable oder ThrowingSupplier.

Listing 23.6     com/tutego/insel/junit/util/StringsTest.java, reverse_execution_time_below_1ms()

@Test

void reverse_execution_time_below_1ms() {

assertTimeout( Duration.ofMillis(1), () -> {

Strings.reverse( "abc" );

} );

}

Wird die Testmethode dann nicht innerhalb der Schranke ausgeführt, gilt das als fehlgeschlagener Test, und JUnit meldet einen Fehler.

 

Zum Seitenanfang

23.2.7    Beschriftungen mit @DisplayName Zur vorigen ÜberschriftZur nächsten Überschrift

Wir haben einen Methodennamen zum Beispiel reverse_non_null_string genannt, damit später bei der Testausführung der Name besser zu lesen ist. Tests können aber für die Ausgabe mit @DisplayName eine eigene Beschriftung bekommen:

@Test

@DisplayName( "reverse a non null string" )

void reverseNonNullString() { ... }

So lässt sich, wenn gewünscht, ein Methodenname nach der üblichen Namenskonvention einsetzen.

 

Zum Seitenanfang

23.2.8    Verschachtelte Tests Zur vorigen ÜberschriftZur nächsten Überschrift

Wählen Programmierer den Ansatz, für jede zu testende Methode nur genau eine @Test-Methode zu schreiben, führt das zu einer großen Anzahl von assert*(…)-Methoden für unterschiedliche Bereiche. Das ist unübersichtlich und nicht optimal, da bei einem Fehler die ganze Testmethode abbricht.

Eine gute Lösung, um Tests zu einem Vorgang oder einer zu testenden Methode zu bündeln, sind verschachtelte Tests. JUnit setzt sie mit geschachtelten Klassen um, die mit @Nested annotiert sind. Die Annotation @DisplayName ist nicht zwingend, aber sehr praktisch:

Listing 23.7     com/tutego/insel/junit/util/StringsTest.java, Ausschnitt

class StringsTest {



@DisplayName( "reverse(string)" )

@Nested class reverse {



@Test void reverse_non_null_string() { ... }



@Test void reverse_null_string_2() { ... }

}

}

Sollte es in der Utility-Klasse neben reverse(String) noch weitere Methoden geben, so könnten jeweils dafür geschachtelte Klassen existieren. Prinzipiell könnten die Ebenen beliebig tief gehen.

 

Zum Seitenanfang

23.2.9    Tests ignorieren Zur vorigen ÜberschriftZur nächsten Überschrift

Durch Umstrukturierung von Quellcode kann es sein, dass Testcode nicht länger gültig ist und entfernt oder umgebaut werden muss. Damit der Testfall nicht ausgeführt wird, muss er nicht auskommentiert werden (das bringt den Nachteil mit sich, dass sich das Refactoring etwa im Zuge einer Umbenennung von Bezeichnern nicht auf auskommentierte Bereiche auswirkt). Stattdessen lässt sich eine weitere Annotation @Disabled an die Methode setzen:

@Disabled @Test

void reverse_non_null_string()
 

Zum Seitenanfang

23.2.10    Mit Methoden der Assumptions-Klasse Tests abbrechen Zur vorigen ÜberschriftZur nächsten Überschrift

Während die assert*(…)-Methoden bei einem Fehlschlag intern zu einer Ausnahme führen und so anzeigen, dass der Test etwas gefunden hat, was nicht korrekt ist, bietet JUnit mit Assumptions.assume*(…)-Methoden die Möglichkeit, die Tests nicht fortzuführen. Das ist zum Beispiel dann sinnvoll, wenn die Testausführung nicht möglich ist, etwa weil der Testrechner keine Grafikkarte hat, das Netzwerk nicht reagiert oder das Datensystem voll ist. Dabei geht es nicht darum, zu testen, wie sich die Routine bei einem nicht vorhandenen Netzwerk verhält – das gilt es natürlich auch zu testen. Aber steht das Netzwerk nicht, dann können logischerweise auch keine Tests laufen, die das Netzwerk zwingend benötigen. Zwei der über 10 Methoden sind:

class org.junit.jupiter.api.Assumptions
  • static void assumeTrue(boolean assumption)

  • static void assumeFalse(boolean assumption)

Die assume*(…)-Methoden führen zu keiner Ausnahme, brechen die Testausführung aber ab.

 

Zum Seitenanfang

23.2.11    Parametrisierte Tests Zur vorigen ÜberschriftZur nächsten Überschrift

Bei Testfällen werden oft die zu testenden Methoden mit unterschiedlichen Werten gefüttert. Wir hatten so einen Fall:

assertEquals( "",    Strings.reverse( "" ) );

assertEquals( "cba", Strings.reverse( "abc" ) );

Das riecht nach Codeduplikation, die sich mit parametrisierten Tests reduzieren lässt.

org.junit.jupiter:junit-jupiter-params Dependency

Parametrisierte Tests sind kein Teil des JUnit-Kerns, sodass wir eine neue Abhängigkeit in der Maven-POM einfügen müssen:

Listing 23.8     pom.xml, Ausschnitt

<dependency>

<groupId>org.junit.jupiter</groupId>

<artifactId>junit-jupiter-params</artifactId>

<version>5.8.1</version>

<scope>test</scope>

</dependency>

@org.junit.jupiter.params.ParameterizedTest

Gehen wir zurück zum Java-Code. Für parametrisierte Tests werden die Methoden nicht mehr mit @org.junit.jupiter.api.Test annotiert, sondern stattdessen mit @org.junit.jupiter.params.ParameterizedTest.

Als Nächstes müssen gültige Eingabewerte bestimmt werden, und hierfür gibt es mehrere Möglichkeiten; zwei wollen wir uns anschauen.

Als Erstes besteht die Möglichkeit, über @ValueSource eine Sammlung von Werten vorzugeben. Diverse Datentypen sind möglich, darunter numerische Werte, Strings und Class-Objekte. JUnit läuft die Sammlung ab und übermittelt jeden Wert über den Methodenparameter an die Testmethode. Wir können dann diese Parameter verarbeiten und zum Beispiel an die zu testende Methode übergeben:

Listing 23.9     com/tutego/insel/junit/util/StringsTest.java, Ausschnitt

@ParameterizedTest

@ValueSource( strings = { "", " ", "abc" } )

void reverse_will_not_throw_exception_with_non_null_inputs( String input ) {

Strings.reverse( input );

}

Was wir mit @ValueSource nicht machen können, ist, neben den gegebenen Werten auch die erwarteten Ergebnisse mit zu übermitteln. Das ermöglicht stattdessen @CsvSource:

Listing 23.10     com/tutego/insel/junit/util/StringsTest.java, Ausschnitt

@ParameterizedTest

@CsvSource( { "a,a", "ab,ba", "abc,cba" } )

void reverse_non_null_inputs( String input, String expected ) {

assertEquals( expected, Strings.reverse( input ) );

}

Achtung: Ein »,« für den leeren String können wir nicht nehmen, sonst folgt eine IllegalArgumentException. Standardmäßig ist der Delimiter ein Komma, aber das kann geändert werden.

 


Ihre Meinung?

Wie hat Ihnen das Openbook gefallen? Wir freuen uns immer über Ihre Rückmeldung. Schreiben Sie uns gerne Ihr Feedback als E-Mail an kommunikation@rheinwerk-verlag.de

<< zurück
 Zum Rheinwerk-Shop
Zum Rheinwerk-Shop: Java ist auch eine Insel Java ist auch eine Insel

Jetzt Buch bestellen


 Buchempfehlungen
Zum Rheinwerk-Shop: Captain CiaoCiao erobert Java

Captain CiaoCiao erobert Java




Zum Rheinwerk-Shop: Algorithmen in Java

Algorithmen in Java




Zum Rheinwerk-Shop: Spring Boot 3 und Spring Framework 6

Spring Boot 3 und Spring Framework 6




Zum Rheinwerk-Shop: Java SE 9 Standard-Bibliothek

Java SE 9 Standard-Bibliothek




 Lieferung
Versandkostenfrei bestellen in Deutschland, Österreich und in die Schweiz

InfoInfo



 

 


Copyright © Rheinwerk Verlag GmbH 2024

Für Ihren privaten Gebrauch dürfen Sie die Online-Version natürlich ausdrucken. Ansonsten unterliegt das Openbook denselben Bestimmungen, wie die gebundene Ausgabe: Das Werk einschließlich aller seiner Teile ist urheberrechtlich geschützt.

Alle Rechte vorbehalten einschließlich der Vervielfältigung, Übersetzung, Mikroverfilmung sowie Einspeicherung und Verarbeitung in elektronischen Systemen.

 

[Rheinwerk Computing]



Rheinwerk Verlag GmbH, Rheinwerkallee 4, 53227 Bonn, Tel.: 0228.42150.0, Fax 0228.42150.77, service@rheinwerk-verlag.de



Cookie-Einstellungen ändern