Rheinwerk Computing < openbook > Rheinwerk Computing - Professionelle Bücher. Auch für Einsteiger.
Professionelle Bücher. Auch für Einsteiger.
 
Inhaltsverzeichnis
Vorwort
1 Neues in Java 8 und Java 7
2 Fortgeschrittene String-Verarbeitung
3 Threads und nebenläufige Programmierung
4 Datenstrukturen und Algorithmen
5 Raum und Zeit
6 Dateien, Verzeichnisse und Dateizugriffe
7 Datenströme
8 Die eXtensible Markup Language (XML)
9 Dateiformate
10 Grafische Oberflächen mit Swing
11 Grafikprogrammierung
12 JavaFX
13 Netzwerkprogrammierung
14 Verteilte Programmierung mit RMI
15 RESTful und SOAP-Web-Services
16 Technologien für die Infrastruktur
17 Typen, Reflection und Annotationen
18 Dynamische Übersetzung und Skriptsprachen
19 Logging und Monitoring
20 Sicherheitskonzepte
21 Datenbankmanagement mit JDBC
22 Java Native Interface (JNI)
23 Dienstprogramme für die Java-Umgebung
Stichwortverzeichnis

Jetzt Buch bestellen
Ihre Meinung?

Spacer
<< zurück
Java SE 8 Standard-Bibliothek von Christian Ullenboom
Das Handbuch für Java-Entwickler
Buch: Java SE 8 Standard-Bibliothek

Java SE 8 Standard-Bibliothek
Pfeil 7 Datenströme
Pfeil 7.1 Stream-Klassen für Bytes und Zeichen
Pfeil 7.1.1 Lesen aus Dateien und Schreiben in Dateien
Pfeil 7.1.2 Byteorientierte Datenströme über Files beziehen
Pfeil 7.1.3 Zeichenorientierte Datenströme über Files beziehen
Pfeil 7.1.4 Funktion von OpenOption bei den Files.newXXX(…)-Methoden
Pfeil 7.1.5 Ressourcen aus dem Klassenpfad und aus JAR‐Archiven laden
Pfeil 7.1.6 Die Schnittstellen Closeable, AutoCloseable und Flushable
Pfeil 7.2 Basisklassen für die Ein-/Ausgabe
Pfeil 7.2.1 Die abstrakten Basisklassen
Pfeil 7.2.2 Übersicht über Ein-/Ausgabeklassen
Pfeil 7.2.3 Die abstrakte Basisklasse OutputStream
Pfeil 7.2.4 Ein Datenschlucker *
Pfeil 7.2.5 Die abstrakte Basisklasse InputStream
Pfeil 7.2.6 Ströme mit SequenceInputStream zusammensetzen *
Pfeil 7.2.7 Die abstrakte Basisklasse Writer
Pfeil 7.2.8 Die Schnittstelle Appendable *
Pfeil 7.2.9 Die abstrakte Basisklasse Reader
Pfeil 7.3 Formatierte Textausgaben
Pfeil 7.3.1 Die Klassen PrintWriter und PrintStream
Pfeil 7.3.2 System.out, System.err und System.in
Pfeil 7.4 Die FileXXX-Stromklassen
Pfeil 7.4.1 Kopieren mit FileOutputStream und FileInputStream
Pfeil 7.4.2 Das FileDescriptor-Objekt *
Pfeil 7.4.3 Mit dem FileWriter Texte in Dateien schreiben
Pfeil 7.4.4 Zeichen mit der Klasse FileReader lesen
Pfeil 7.5 Schreiben und Lesen aus Strings und Byte-Feldern
Pfeil 7.5.1 Mit dem StringWriter ein String-Objekt füllen
Pfeil 7.5.2 CharArrayWriter
Pfeil 7.5.3 StringReader und CharArrayReader
Pfeil 7.5.4 Mit ByteArrayOutputStream in ein Byte-Feld schreiben
Pfeil 7.5.5 Mit ByteArrayInputStream aus einem Byte-Feld lesen
Pfeil 7.6 Datenströme filtern und verketten
Pfeil 7.6.1 Streams als Filter verketten (verschachteln)
Pfeil 7.6.2 Gepufferte Ausgaben mit BufferedWriter und BufferedOutputStream
Pfeil 7.6.3 Gepufferte Eingaben mit BufferedReader/BufferedInputStream
Pfeil 7.6.4 LineNumberReader zählt automatisch Zeilen mit *
Pfeil 7.6.5 Daten mit der Klasse PushbackReader zurücklegen *
Pfeil 7.6.6 DataOutputStream/DataInputStream *
Pfeil 7.6.7 Basisklassen für Filter *
Pfeil 7.6.8 Die Basisklasse FilterWriter *
Pfeil 7.6.9 Ein LowerCaseWriter *
Pfeil 7.6.10 Eingaben mit der Klasse FilterReader filtern *
Pfeil 7.6.11 Anwendungen für FilterReader und FilterWriter *
Pfeil 7.7 Vermittler zwischen Byte-Streams und Unicode-Strömen
Pfeil 7.7.1 Datenkonvertierung durch den OutputStreamWriter
Pfeil 7.7.2 Automatische Konvertierungen mit dem InputStreamReader
Pfeil 7.8 Kommunikation zwischen Threads mit Pipes *
Pfeil 7.8.1 PipedOutputStream und PipedInputStream
Pfeil 7.8.2 PipedWriter und PipedReader
Pfeil 7.9 Prüfsummen
Pfeil 7.9.1 Die Schnittstelle Checksum
Pfeil 7.9.2 Die Klasse CRC32
Pfeil 7.9.3 Die Adler32-Klasse
Pfeil 7.10 Persistente Objekte und Serialisierung
Pfeil 7.10.1 Objekte mit der Standardserialisierung speichern und lesen
Pfeil 7.10.2 Zwei einfache Anwendungen der Serialisierung *
Pfeil 7.10.3 Die Schnittstelle Serializable
Pfeil 7.10.4 Nicht serialisierbare Attribute aussparen
Pfeil 7.10.5 Das Abspeichern selbst in die Hand nehmen
Pfeil 7.10.6 Tiefe Objektkopien *
Pfeil 7.10.7 Versionenverwaltung und die SUID
Pfeil 7.10.8 Wie die ArrayList serialisiert *
Pfeil 7.10.9 Probleme mit der Serialisierung
Pfeil 7.11 Alternative Datenaustauschformate
Pfeil 7.11.1 Serialisieren in XML-Dateien
Pfeil 7.11.2 XML-Serialisierung von JavaBeans mit JavaBeans Persistence *
Pfeil 7.11.3 Die Open-Source-Bibliothek XStream *
Pfeil 7.11.4 Binäre Serialisierung mit Google Protocol Buffers *
Pfeil 7.12 Zum Weiterlesen
 
Zum Seitenanfang

7.3Formatierte Textausgaben Zur vorigen ÜberschriftZur nächsten Überschrift

Um mit einem Writer oder OutputStream komfortabel Zeichenfolgen – zum Beispiel String-Repräsentationen primitiver Datentypen – zu erzeugen, ist eine Menge Schreibaufwand nötig, da ein einfacher Writer nur Zeichen bzw. Strings und ein simpler OutputStream sogar nur Bytes oder Byte-Folgen schreibt. Dieser Abschnitt stellt die beiden Klassen PrintWriter und PrintStream vor, die das Erzeugen textueller Ausgaben vereinfachen und das Ergebnis in einen Zielstrom oder in eine Datei schreiben.

 
Zum Seitenanfang

7.3.1Die Klassen PrintWriter und PrintStream Zur vorigen ÜberschriftZur nächsten Überschrift

Schon in den ersten Programmen haben wir ein PrintStream-Objekt verwendet – doch vermutlich, ohne es zu wissen. Es steckte im out-Attribut der Klasse System. Typisch für PrintStream und PrintWriter sind die vielen überladenen Methoden namens print(…), println(…), printf(…) und format(…), die unterschiedliche Datentypen annehmen und eine String-Repräsentation erzeugen. Anders als viele Stromklassen erzeugen die Klassen bei Schreibproblemen keine IOException, sondern setzen intern ein Flag, das durch die Methode checkError() nach außen kommt.

UML-Diagramme der Klassen PrintStream und PrintWriter

Abbildung 7.2UML-Diagramme der Klassen PrintStream und PrintWriter

[zB]Beispiel

Schreibe eine Zeichenkette und die String-Repräsentation des aktuellen Datums in eine Datei:

Listing 7.11com/tutego/insel/io/stream/WriteInFile.java, main()

try ( PrintWriter out = new PrintWriter( "c:/datei.txt",
StandardCharsets.ISO_8859_1.name() ) ) {
out.println( "Hallo Welt!" );
out.print( "Es ist " );
out.printf( "%tT Uhr.", new Date() );
}
catch ( FileNotFoundException | UnsupportedEncodingException e ) {
e.printStackTrace();
}

Obwohl die Methoden selbst keine Ausnahmen auslösen können, kann es doch dieser spezielle Konstruktor, der in eine Datei schreibt.

Konstruktoren

Die Konstruktoren von PrintStream und PrintWriter erwarten entweder ein Objekt vom Typ OutputStream, Writer (nur PrintWriter), File oder einen String für den Namen der Datei, in die die Ausgabe geschrieben wird. Mit den beiden zuletzt genannten Konstruktoren wird der PrintWriter/PrintStream nicht als Dekorator um einen existierenden Writer/Stream eingesetzt; es muss also kein FileWriter in den Konstruktor von PrintWriter bzw. kein FileOutputStream im Konstruktor von PrintStream gesetzt werden, um in eine Datei zu schreiben. Technisch gesehen ist ein PrintStream eine Unterklasse von FilterOutputStream (interessanterweise gilt das für die Klasse PrintWriter nicht, die eine direkte Unterklasse von Writer ist.)

class java.io.PrintWriter class java.io.PrintStream
extends Writer extends FilterOutputStream
implements Appendable, Closeable
  • PrintWriter(File file) throws FileNotFoundException

  • PrintStream(File file) throws FileNotFoundException

  • PrintWriter(String fileName)

  • PrintStream(String fileName)

  • PrintWriter(Writer out)

  • PrintWriter(OutputStream out)

  • PrintStream(OutputStream out)
    Erzeugt einen neuen PrintWriter bzw. PrintStream, der in die angegebene Datei – die entweder durch den Dateinamen oder ein File-Objekt gegeben ist –, den Writer bzw. OutputStream schreibt. Bei einem Zeilenende wird nicht automatisch in den Puffer geschrieben.

Auto-Flush

Die Ausgabe in den PrintWriter und PrintStream ist so lange gepuffert, bis flush() ausgeführt wird. Das Verhalten lässt sich mit einem Schalter ändern, denn einige Konstruktoren nehmen optional ein »Auto-Flush« an, das bestimmt, ob die Daten bei einem println(…) oder einem printf(…) bzw. format(…) mit explizitem »%n« aus dem Puffer gespült werden. So schreibt println("Hallo") auf jeden Fall den String in den Datenstrom, wenn der Konstruktor mit autoFlush gleich true aufgerufen wurde, und es bleiben keine Daten im Puffer. Für den PrintStream gilt zusätzlich beim aktivierten Auto-Flush, dass bei jedem Byte »\n« in der Zeichenkette der Puffer geleert wird, also auch bei Zeichen, die über die write(…)-Methoden geschrieben werden.

class java.io.PrintWriter class java.io.PrintStream
extends Writer extends FilterOutputStream
implements Appendable, Closeable
  • PrintWriter(Writer out, boolean autoFlush)

  • PrintWriter(OutputStream out, boolean autoFlush)

  • PrintStream(OutputStream out, boolean autoFlush)
    Erzeugt einen neuen PrintWriter bzw. PrintStream, der automatisch bei autoFlush gleich true auf ein »\n« im Strom achtet und im Puffer befindliche Daten wegschreibt.

Zeichenkodierung und Konvertierungen

Standardmäßig nutzt der PrintWriter die voreingestellte Zeichenkodierung der Plattform, um Unicodezeichen anzunehmen und sie als Byte weiterzureichen. Eine explizite Kodierung lässt sich angeben.

class java.io.PrintWriter class java.io.PrintStream
extends Writer extends FilterOutputStream
implements Appendable, Closeable
  • PrintWriter(File file, String csn)
    throws FileNotFoundException, UnsupportedEncodingException

  • PrintStream(File file, String csn)
    throws FileNotFoundException, UnsupportedEncodingException

  • PrintWriter(String fileName, String csn)
    throws FileNotFoundException, UnsupportedEncodingException

  • PrintStream(String fileName, String csn)
    throws FileNotFoundException, UnsupportedEncodingException
    Erzeugt einen neuen PrintWriter zum Schreiben in Dateien mit den angegebenen Kodierungen.

  • PrintStream(OutputStream out, boolean autoFlush, String encoding)
    throws UnsupportedEncodingException
    Erzeugt einen PrintStream mit gewünschter Pufferleerung und Zeichenkodierung.

[»]Hinweis

Die Zeichenkodierung sollte immer explizit angegeben werden!

Methodenübersicht

Als Methoden bieten PrintWriter und PrintStream im Wesentlichen das Gleiche: eine Reihe von print(…)-, println(…)-, format(…)-, printf(…)- sowie append(…)-Methoden aus Appendable.

class java.io.PrintWriter class java.io.PrintStream
extends Writer extends FilterOutputStream
implements Appendable, Closeable
  • void print(boolean b)

  • void print(char c)

  • void print(char[] s)

  • void print(double d)

  • void print(float f)

  • void print(int i)

  • void print(String s)
    Schreibt boolean, ein einzelnes Zeichen, ein Array von Zeichen, Fließkommazahlen, Ganzzahlen oder einen String. Ist das char-Feld null, gibt es eine NullPointerException.

  • void print(Object o)
    Ruft o.toString() auf und gibt das Ergebnis aus, wenn o ungleich null ist. Sonst ist die Ausgabe null.

  • void println()
    Schreibt Zeilenvorschubzeichen.

  • void println(boolean b)

  • void println(char c)

  • void println(char[] s)

  • void println(double d)

  • void println(float f)

  • void println(int i)

  • void println(String s)
    Schreibt den Datentyp wie print(…) und schließt die Zeile mit einem Zeilenendezeichen ab.

  • PrintWriter/PrintStream format(Locale l, String format, Object... args)

  • PrintWriter/PrintStream format(String format, Object... args)

  • PrintWriter/PrintStream printf(Locale l, String format, Object... args)

  • PrintWriter/PrintStream printf(String format, Object... args)
    Gibt eine nach dem Formatierungs-String gegebene Zeichenfolge aus. Ist das Format null, folgt eine NullPointerException; bei inkorrektem Formatierungs-String folgt eine IllegalFormatException.

  • PrintWriter/PrintStream append(char c)

  • PrintWriter/PrintStream append(CharSequence csq)

  • PrintWriter/PrintStream append(CharSequence csq, int start, int end)
    Schreibt Zeichen, Felder von Zeichen oder einen String. Die append(…)-Methoden stammen aus der Schnittstelle Appendable.

  • void close()
    Schließt den Stream. Ist eine Methode aus Closeable.

  • void flush()
    Schreibt gepufferte Daten. Ist eine Methode aus Flushable.

Der PrintWriter und PrintStream implementieren als typischer Writer und OutputStream noch die write(…)-Methoden; einmal mit char und einmal mit byte.

Umgang im Fehlerfall

Keine der Methoden löst eine IOException aus (nur die Konstruktoren können eine FileNotFoundException, also eine IOException auslösen). Intern fangen der PrintWriter und PrintStream eine mögliche IOException ab und setzen ein internes Flag trouble, auf das der Programmablauf jedoch keinen weiteren Einfluss hat. (Daher musste auch die Methode write(String) neu deklariert werden, da die Methode write(String) der Writer-Klasse eine IOException auslöst.) Den Fehlerstatus erfragt checkError().

class java.io.PrintWriter class java.io.PrintStream
extends Writer extends FilterOutputStream
implements Appendable, Closeable
  • boolean checkError()
    Schreibt mit flush() gepufferte Daten und testet, ob intern bei der Verarbeitung eine IOException (außer InterruptedIOException) aufgetreten ist oder manuell setError() aufgerufen wurde. Die Abfrage ist wichtig, da die Klasse keine Ein-/Ausgabe-Exception auslöst.

  • protected void setError()
    Setzt den Fehlerstatus auf true. Mit der protected-Methode können Unterklassen eines PrintWriter den Fehlerstatus setzen, den checkError() einbezieht.

  • protected void clearError()
    Löscht den internen Fehlerstatus.

 
Zum Seitenanfang

7.3.2System.out, System.err und System.in Zur vorigen ÜberschriftZur nächsten Überschrift

Beim Aufbau der System-Klasse erzeugt Java automatisch drei Stream-Objekte, die üblicherweise die Eingaben über die Tastatur und Ausgaben über den Bildschirm repräsentieren:

  • System.in ist das Standardeingabegerät. System.in ist ein Exemplar der Klasse InputStream (genauer gesagt vom Typ BufferedInputStream).

  • System.out und System.err stehen für das Standardausgabegerät für normale Ausgaben und Fehlerausgaben. Sie sind Exemplare von PrintStream.

[»]Hinweis

Wird ein Programm von der Konsole aus aufgerufen, so können mithilfe so genannter Umleitungen auch Datenströme von Dateien in das (und aus dem) Java-Programm gelenkt werden, sodass dann System.in und System.out/System.err automatisch Dateiströme sind und somit nicht mehr auf die Konsole gehen. Das kann etwa so aussehen:

$ java WithRedirectedStreams > out.text < in.txt
final class java.lang.System
  • static final InputStream in
    Dies ist der Standardeingabestrom. Er ist immer geöffnet und nimmt die Benutzereingaben normalerweise über die Tastatur entgegen.

  • static final PrintStream out
    Der Standardausgabestrom. Er ist immer geöffnet und normalerweise mit der Bildschirmausgabe verbunden.

  • static final PrintStream err
    Der Standard-Fehlerausgabestrom. Er wurde eingeführt, um die Fehlermeldungen von den Ausgabemeldungen zu unterscheiden. Auch wenn der Ausgabekanal umgeleitet wird, bleiben diese Meldungen erreichbar.

[zB]Beispiel

In eigenen Projekten gibt es selten die Notwendigkeit für PrintStream, da die Bytes nicht in einer speziellen Kodierung geschrieben werden. Wir stützen uns auf die Klasse PrintWriter, die die abstrakte Klasse Writer erweitert. Das Attribut System.out bleibt weiterhin vom Typ PrintStream, und die Deklaration PrintWriter o = System.out ist falsch.

Wenn die Standardkodierung in Ordnung ist, kann PrintStream für Debug-Code auf die Konsole mittels System.out verwendet werden. Sonst ist PrintWriter eine bessere Wahl, da er auch nach System.out schreiben kann:

PrintWriter o = new PrintWriter( System.out );

Schreibarbeit sparen *

Natürlich ist es Schreibarbeit, immer System.out.printXXX(…) angeben zu müssen, so wie in:

System.out.println( "Das Programm gibt die Ausgabe:" );
System.out.println( 1.234 );
System.out.println( "Die drei Fragezeichen sind toll." );

Eine Möglichkeit, das Ganze abzukürzen, besteht darin, das statische import zu nutzen. Doch auch mit einem Verweis können wir uns Arbeit sparen. Das funktioniert, da System.out ein Objekt vom Typ PrintStream ist:

final PrintStream o = System.out;
o.println( "Neu!" );
o.println( "Jetzt noch weniger zu schreiben." );
o.println( "Hilft auch Gelenken wieder auf die Sprünge!" );

Ströme umlenken *

Die Ströme hinter System.out, System.err und System.in sind vom Laufzeitsystem vorbelegt und lassen sich ändern. Dazu deklariert die Klasse System drei statische Methoden:

final class java.lang.System
  • static void setOut(PrintStream out)

  • static void setErr(PrintStream err)
    Der Standardausgabekanal/Fehlerkanal wird auf den angegebenen PrintStream gesetzt.

  • static void setIn(InputStream in)
    Der Eingabestrom kann umgesetzt werden, um beispielsweise aus einer Datei oder Netzwerkverbindung Daten zu beziehen, die an in anliegen sollen.

[zB]Beispiel

Setze den Eingabestrom auf einen Latin-1-String mit drei Zeilen:

System.setIn( new ByteArrayInputStream(
"1\n2\n3\n".getBytes(StandardCharsets.ISO_8859_1) ) );

Für Applikationen ist es nur möglich, über die drei genannten Methoden die Standardeingabe auf einen beliebigen InputStream und die Standardausgabe auf einen beliebigen PrintStream umzuleiten. Sehr erstaunlich in der System-Klasse ist die Tatsache, dass die Attribute in, out und err final sind und daher eigentlich nicht geändert werden können. Hier greift das JDK tief in die Trickkiste.

Den Bildschirm löschen und Textausgaben optisch aufwerten *

Die Java-Umgebung setzt keine spezielle grafische Architektur und kein spezielles Terminal voraus. Als systemunabhängige Programmiersprache bietet Java außer der einfachen Textausgabe keine Möglichkeit, die Farben für die Textzeichen zu ändern, den Cursor zu setzen oder den Bildschirm zu löschen.

Ein VT100-Terminal speichert neben dem Text auch Attribute, und es existieren unterschiedliche Kontrollsequenzen, sodass die Konsole Text formatiert ausgeben kann. Wir können bei einem VT100-Terminal Kontrollzeichen über System.out ausgeben und so zum Beispiel den Bildschirm löschen.

System.out.println( "\u001b[H\u001b[2J" );

Diese Kontrollzeichen sind kryptisch, und so gibt es eine Curses-Bibliothek, mit der Entwickler auf einem VT100-Terminal Zeichenattribute auf der Konsole ändern, den Cursor zum Wandern bringen und einfache ASCII-Fenster erstellen können. Unter Unix implementiert ncurses die Funktionalität – unter Windows gibt es Vergleichbares als PDCurses. Zwei bekanntere Open-Source-Bibliotheken bieten Java-Zugriff auf die Curses-Bibliothek: CHARVA (http://www.pitman.co.za/projects/charva/) und Jcurzez (http://www.nongnu.org/jcurzez/).

 


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 SE 8 Standard-Bibliothek Java SE 8 Standard-Bibliothek
Jetzt Buch bestellen

 Buchempfehlungen
Zum Rheinwerk-Shop: Java ist auch eine Insel
Java ist auch eine Insel


Zum Rheinwerk-Shop: Professionell entwickeln mit Java EE 8
Professionell entwickeln mit Java EE 8


Zum Rheinwerk-Shop: Besser coden
Besser coden


Zum Rheinwerk-Shop: Entwurfsmuster
Entwurfsmuster


Zum Rheinwerk-Shop: IT-Projektmanagement
IT-Projektmanagement


 Lieferung
Versandkostenfrei bestellen in Deutschland, Österreich und der Schweiz
InfoInfo

 
 


Copyright © Rheinwerk Verlag GmbH 2018
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.

 
Nutzungsbestimmungen | Datenschutz | Impressum

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

Cookie-Einstellungen ändern