Datei effizient kopieren

Status
Nicht offen für weitere Antworten.

Noar

Bekanntes Mitglied
Hallo,

ich habe eine Frage zu einer Studienaufgabe:

Die Aufgabe ist, eine Datei effektiv zu kopieren.
Randbedingungen: Datei kann Byte für Byte kopiert werden. Dateiname/Pfad soll vom Nutzer auf der Konsole eingegeben werden können.
Der Name der Zieldatei ist mit der Quelldatei identisch, nur um ein .bak ergänzt.
Nutzer soll über Erfolg/Misserfolg unterrichtet werde.

Ich habe das mal umgesetzt und es funktioniert. Könnt ihr mal drüberschauen, und mir sagen, ob es da noch was zu verbessern gibt (eleganz, effizienz etc.) Bitte auch Tips geben, wenn ich Konventionen etc. verletze.


Code:
import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;

public class DateiKopierer {

	/**
	 * This class copies a file byte by byte.
	 * The original filename will be reused (with a new suffix "bak").
	 * 
	 * @param args
	 * @author Thomas Wenzel
	 */
	public static void main(String[] args) {
		
		//Read filename from console
		BufferedReader buffReader=new BufferedReader(new InputStreamReader(System.in));
		String strInput;
		try {		
			System.out.print("Source file: ");
        	strInput = buffReader.readLine();
        	
        	//Copy file
        	String strDestFilename = strInput+".bak";
            File fileSrc = new File(strInput);
    		File fileDes = new File(strDestFilename);
    		System.out.println("Copy file '"+strInput+"' to '"+strDestFilename+"'");
            try {
            	byte[] streamBuffer  = new byte[1024]; // Buffer für gelesene Daten
        		InputStream in = new FileInputStream(fileSrc);
        		OutputStream out = new FileOutputStream(fileDes);
        		
        		for (int n; (n = in.read(streamBuffer)) != -1; out.write(streamBuffer, 0 , n));
        		in.close();
        		out.flush(); 
        		out.close();
        		System.out.println("Copy successfully finished.");
        			
    		} catch (IOException e) {
    			System.out.println("Error while copy file '"+strInput+"'");
    			e.printStackTrace();
    		}
    		
		}
        catch(IOException e) {
            System.out.println("Input data error!");
        } 
		
	}
}

Vielen Dank für eure Hilfe
Noar
 
G

Guest

Gast
Noar hat gesagt.:
Hallo,

ich habe eine Frage zu einer Studienaufgabe:

Die Aufgabe ist, eine Datei effektiv zu kopieren.
Randbedingungen: Datei kann Byte für Byte kopiert werden. Dateiname/Pfad soll vom Nutzer auf der Konsole eingegeben werden können.
Der Name der Zieldatei ist mit der Quelldatei identisch, nur um ein .bak ergänzt.
Nutzer soll über Erfolg/Misserfolg unterrichtet werde.

Ich habe das mal umgesetzt und es funktioniert. Könnt ihr mal drüberschauen, und mir sagen, ob es da noch was zu verbessern gibt (eleganz, effizienz etc.) Bitte auch Tips geben, wenn ich Konventionen etc. verletze.


Code:
import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;

public class DateiKopierer {

	/**
	 * This class copies a file byte by byte.
	 * The original filename will be reused (with a new suffix "bak").
	 * 
	 * @param args
	 * @author Thomas Wenzel
	 */
	public static void main(String[] args) {
		
		//Read filename from console
		BufferedReader buffReader=new BufferedReader(new InputStreamReader(System.in));
		String strInput;
		try {		
			System.out.print("Source file: ");
        	strInput = buffReader.readLine();
        	
        	//Copy file
        	String strDestFilename = strInput+".bak";
            File fileSrc = new File(strInput);
    		File fileDes = new File(strDestFilename);
    		System.out.println("Copy file '"+strInput+"' to '"+strDestFilename+"'");
            try {
            	byte[] streamBuffer  = new byte[1024]; // Buffer für gelesene Daten
        		InputStream in = new FileInputStream(fileSrc);
        		OutputStream out = new FileOutputStream(fileDes);
        		
        		for (int n; (n = in.read(streamBuffer)) != -1; out.write(streamBuffer, 0 , n));
        		in.close();
        		out.flush(); 
        		out.close();
        		System.out.println("Copy successfully finished.");
        			
    		} catch (IOException e) {
    			System.out.println("Error while copy file '"+strInput+"'");
    			e.printStackTrace();
    		}
    		
		}
        catch(IOException e) {
            System.out.println("Input data error!");
        } 
		
	}
}

Vielen Dank für eure Hilfe
Noar

Also ich finde das Kopieren sieht doch schon ganz gut aus. Auch die Codeformatierung ist gut so (sind nur n paar Verschieber drin).

Was die Effizienz des Kopierens angeht ists immer schwer zu sagen. Wenn ich z. B. keine Verlaufsanzeige oder sonstiges brauche benutze ich auch oft mal den FileChannel zum Kopieren. Ich weiß einige hier im Forum sind nicht so begeistert vom FileChannel, aber ich hatte bis jetzt keine Probleme damit.

Ein Kopieren mit dem FileChannel könnte dann so aussehen:

Code:
   private void copy(final File from, final File to) {

      try {
         final FileChannel in = new FileInputStream(from).getChannel();
         final FileChannel out = new FileOutputStream(to).getChannel();

         out.transferFrom(in, 0, in.size());

         in.close();
         out.close();

         System.out.println(from.getPath() + " - erfolgreich kopiert");
      } catch (FileNotFoundException fnfe) {
         System.out.println("Datei nicht gefunden: " + from.getPath());
         fnfe.printStackTrace();
      } catch (IOException ioe) {
         System.out.println("IOException: " + from.getPath());
         ioe.printStackTrace();
      }
   }
 

norman

Top Contributor
Hallo Noar,

Code:
for (int n; (n = in.read(streamBuffer)) != -1; out.write(streamBuffer, 0 , n));
finde ich ziemlich umständlich zu lesen..aber finde es immer krass, was man mit der for-schleife so alles machen kann (oft werden ja auch anweisungen weggelassen ( for (;bla;blubb)..)

ansonsten kann es evtl sinnvoll sein den dateinamen/ pfad gleich als parameter beim starten zu übergeben. (strInput = args[0])
 

Bleiglanz

Gesperrter Benutzer
ich würde vielleicht noch die Blockgrösse etwas grösser machen, 1024 ist wahrscheinlich nicht so optimal (viele Betriebssysteme arbeiten mit grösseren Blöcken)

und
Code:
for (int n; (n = in.read(streamBuffer)) != -1; out.write(streamBuffer, 0 , n));
ist zwar superkurz und elegant, aber nicht so toll lesbar
lieber ein paar zeilen mehr schreiben, in Punkto Geschwindigkeit macht das keinen Unterschied

und: mach lieber eine Funktion draus, so a la copy(File src, File dest)
 

SamHotte

Top Contributor
Vielleicht kann man die OS-Blockgröße ja sogar auslesen, das dürfte meistens die schnellsten File-IO-Operationen erlauben, oder?
 

SamHotte

Top Contributor
Keine Ahnung, aber sonstige systemspezifische Dinge kann Java ja auch, wie bspw. in java.awt.Toolkit die Bildschirmauflösung holen. Vielleicht gibt es ja auch Wrapperklassen fürs Dateisystem ...
 

Noar

Bekanntes Mitglied
Hallo,

vielen Dank für eure Antworten.

Ich habe die bemängelte for-Schleife mal entschärft. Ist die Variante mit while-Schleife besser?
Was spricht eigentlich gegen das for-Schleifenkonstukt?

Ich habe auch die Kopieraktion in eine Methode ausgelagert. Besser?

Was die Blockgröße angeht, da habt ihr sicher recht. Wäre natürlich sehr elegant, die an die jeweilige Umgebung anzupassen. Allerdings glaube ich, ist das für meinen momentanen Kenntnisstand noch etwas oversized.

Welchen Wert kann man denn für moderne Systeme nehmen?



Code:
import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;

public class DateiKopierer {

	/**
	 * This class copies a file byte by byte.
	 * The original filename will be reused (with a new suffix "bak").
	 **/
	public static void main(String[] args) {

		//Read filename from console
		BufferedReader buffReader = new BufferedReader(new InputStreamReader(
				System.in));
		String strInput;
		try {
			System.out.print("Source file: ");
			strInput = buffReader.readLine();

			copyFile(strInput); //rufe Methode copyFile auf

		} catch (IOException e) {
			System.out.println("Input data error!");
		}

	}

	public static void copyFile(String strInput) {
		String strDestFilename = strInput + ".bak";
		File fileSrc = new File(strInput);
		File fileDes = new File(strDestFilename);
		System.out.println("Copy file '" + strInput + "' to '"
				+ strDestFilename + "'");
		try {
			byte[] streamBuffer = new byte[1024]; // Buffer für gelesene Daten
			InputStream in = new FileInputStream(fileSrc);
			OutputStream out = new FileOutputStream(fileDes);

			int n;
			while ((n = in.read(streamBuffer)) > 0) {
				out.write(streamBuffer, 0, n);
			}

			in.close();
			out.flush();
			out.close();
			System.out.println("Copy successfully finished.");

		} catch (IOException e) {
			System.out.println("Error while copy file '" + strInput
					+ "'. File not found.");
			//e.printStackTrace();
		}
	}
}


Gruß und Dank
Noar
 

norman

Top Contributor
ich würde es der lesbarkeit wegen sogar so machen:
Code:
       int n=1;
        while (n > 0) {
            n = in.read(streamBuffer)
            out.write(streamBuffer, 0, n);
         }
 

Leroy42

Top Contributor
Bevorzuge ich auch.

Dann müssen die Streams aber auf jeden Fall in ein BufferdInput/OutputStream gewrapped werden
 

Bleiglanz

Gesperrter Benutzer
wegen der for-schleife: es geht uns hier nur um die Lesbarkeit des Programmsquelltextes, für die Geschwindigkeit und alles andere ist das ziemlich wurscht

aber

bitte aufpassen, oben wars richtig:

die bedingung != -1 ist was anders als grösser 0

ist zwar jetzt sehr exotisch (bei Festplatten) aber es kann schon mal vorkommen das keine Bytes gelesen werden können obwohl man noch gar nicht fertig ist
 

norman

Top Contributor
nochmal zu der blockgröße:
in InputStream bin ich gerade auf
Code:
private static final int SKIP_BUFFER_SIZE = 2048;
gestoßen. evtl ist das also ein ganz guter wert..
 

SamHotte

Top Contributor
Und wofür wird die benutzt? Vielleicht nur dann, wenn man im jeweiligen System die "echte" nicht herausfinden kann? AFAIK sind Dateizugriffe mit anderer Größe als der im FS fest verdrahteten immer langsamer, als wenn man genau die dort benutzte nimmt. Und bei OSen != Windows kann man die Blockgröße selbst festlegen, also müsste die JRE diese herausfinden oder automatisch benutzen, oder nicht?
 

Noar

Bekanntes Mitglied
Hallo,
ich danke euch für die Antworten.
Ihr habt mir sehr geholfen.

Gruß und Dank
Noar
 

norman

Top Contributor
keine ahnung wofür die benutzt wird, aber kannst ja nachschauen.. :wink:
vermutlich hast du aber recht. ich wusste nur gar nciht, dass man in einigen OS die blockgröße selbst festlegen kann.
 

SamHotte

Top Contributor
Zumindest unter Linux ging das mal. Die Blockgröße bestimmt die durchschnittliche Platzverschwendung auf deinem Datenträger, denn im Mittel bleibt pro Datei ein halber Block leer.
 
Status
Nicht offen für weitere Antworten.
Ähnliche Java Themen
  Titel Forum Antworten Datum
Z Wie Datei effizient auslesen? Java Basics - Anfänger-Themen 1
K Warum wird hier nur etwas in eine txt Datei geschrieben und nicht in alle drei (InputStream/OutputStream/Reader/Writer) Java Basics - Anfänger-Themen 1
farbenlos Csv Datei in Java einlesen Java Basics - Anfänger-Themen 18
E Audio Datei unter Bedingungen ausführen Java Basics - Anfänger-Themen 19
S Daten aus Import Datei auslesen und sortieren Java Basics - Anfänger-Themen 2
A exe Datei erstellen Java Basics - Anfänger-Themen 8
J .jar datei öffnen funktioniert nicht Java Basics - Anfänger-Themen 17
P Aus Text Datei nur Zahlen übernehmen Java Basics - Anfänger-Themen 13
P Welches SDK für das erstellen einer ausführbaren Datei? Java Basics - Anfänger-Themen 4
W Fehler in der Datei pom.xml Java Basics - Anfänger-Themen 19
M Verständnisfrage: Warum wird die Datei ohne Inhalt übertragen Java Basics - Anfänger-Themen 3
D Jar Datei startet unter Linux nicht Java Basics - Anfänger-Themen 3
P Probleme mit NetBeans: Wie lässt sich jar. Datei an einem MacBook öffnen Java Basics - Anfänger-Themen 21
N Programm Funktioniert mit .txt Datei aber nicht mit .rtf Datei Java Basics - Anfänger-Themen 2
A Wie führe ich eine Batch-Datei von meiner Java-Anwendung aus? Java Basics - Anfänger-Themen 18
D Java Programm mit Batch-Datei starten Java Basics - Anfänger-Themen 32
W Objekte einer ArrayList in txt-datei schreiben mit Paths? Java Basics - Anfänger-Themen 2
E TIF Datei auslesen Java Basics - Anfänger-Themen 2
B von Java/Eclipse verwendete Datei existiert gar nicht? Java Basics - Anfänger-Themen 6
M Spezifischen Wert einer Zeile aus .txt Datei entnehmen Java Basics - Anfänger-Themen 15
B Popups mit Klicksabfangen zumAusfüllen einer .ods Datei Java Basics - Anfänger-Themen 0
M Daten aus .txt Datei einlesen und weiterverarbeiten Java Basics - Anfänger-Themen 80
M RandomAccessFile int und String gleichzeitig in einer Datei Java Basics - Anfänger-Themen 49
I Datei (Bild) Drucken und wie Druckeinstellung speichern? Java Basics - Anfänger-Themen 3
A CSV-Datei Verarbeiten Java Basics - Anfänger-Themen 8
D Downloadfortschritt von Datei über Google Drive API v3 Java Basics - Anfänger-Themen 10
A CSv.Datei einlesen und die werte in zweidemosional Int Array speichern Java Basics - Anfänger-Themen 9
B Den Dateipfad einer Java Datei durch Code in Selbiger finden? Java Basics - Anfänger-Themen 10
S In Datei schreiben in Java? Java Basics - Anfänger-Themen 1
Saiko Zeilen einer Datei einlesen Java Basics - Anfänger-Themen 3
sserio TXT-Datei Auslesen und den Wert jedes Namen ausrechnen etc. Java Basics - Anfänger-Themen 37
sserio Txt Datei einlesen Java Basics - Anfänger-Themen 9
T Printwriter Datei nicht überschreiben Java Basics - Anfänger-Themen 10
berserkerdq2 An selbst ersteller txt Datei immer Text dranhängen, ohne den vorherign Text zu löschen Java Basics - Anfänger-Themen 8
berserkerdq2 Wie gebe ich den Pfad zu einer Datei an, die in einem Ordner in Eclipse ist? Java Basics - Anfänger-Themen 1
D Strings aus Excel-Datei einlesen Java Basics - Anfänger-Themen 2
M Text in Datei schreiben Java Basics - Anfänger-Themen 9
S Datei anlegen Problem! Groß- und Kleinschreibung wird nicht unterschieden Java Basics - Anfänger-Themen 4
J selbst erstellte Datei mit Programm öffnen Java Basics - Anfänger-Themen 10
J int innerhalb einer Datei ändern Java Basics - Anfänger-Themen 1
T208 Text Datei individuell benennen. Java Basics - Anfänger-Themen 5
julian112 Input/Output .gz bzw. .txt Datei Einlesen und Umgang mit Exceptions Java Basics - Anfänger-Themen 1
F Aus eingelesener Datei korrekt Objekte erzeugen Java Basics - Anfänger-Themen 5
E extern Datei von meinem Computer aufmachen Java Basics - Anfänger-Themen 5
H Scripte oder Programmcode aus Datei lesen? Java Basics - Anfänger-Themen 5
E PDF Datei im xfdf-Datei umwandeln und auf dem Laufwerk ablegen Java Basics - Anfänger-Themen 0
J CSV-Datei verarbeiten Java Basics - Anfänger-Themen 27
A Verarbeiten einer Excel Datei durch das java-Programm Java Basics - Anfänger-Themen 3
P Datei einlesen, nach Begriff filtern und in Datei ausgeben. Problem Standardausgabe über Konsole Java Basics - Anfänger-Themen 19
nbergmann Installation unter jdk.java.net: Keine ZIP-Datei zum entpacken Java Basics - Anfänger-Themen 2
J Datei aus Netzwerk auslesen Java Basics - Anfänger-Themen 9
EchtKeineAhnungManchmal hallo habe ein Problem mit einer Datei -> (Zugriff verweigert) Java Basics - Anfänger-Themen 4
EchtKeineAhnungManchmal Controller aus FXML Datei entfernen Java Basics - Anfänger-Themen 49
I Probleme mit OutputStream - Datei lässt sich nicht öffnen Java Basics - Anfänger-Themen 4
Kotelettklopfer Sqlite DB aus Java Datei ansprechen. Java Basics - Anfänger-Themen 147
C XML Datei speichern und laden Java Basics - Anfänger-Themen 18
M Von einem Menü Methode aus anderer Klasse ausführen, die errechnete Werte in Datei schreibt. Java Basics - Anfänger-Themen 8
C XML Datei schreiben Java Basics - Anfänger-Themen 14
S Zufällige ungerade Zeile aus Text-Datei lesen Java Basics - Anfänger-Themen 5
J Wert in einer json Datei ändern und speichern Java Basics - Anfänger-Themen 3
L Java erstellt leere Datei Java Basics - Anfänger-Themen 8
J Json Datei auslesen Java Basics - Anfänger-Themen 4
J In main() Datei geöffnet, von anderer Funktion beschreiben Java Basics - Anfänger-Themen 3
I JAX-WS.... Datei ".ical" zurückgeben.... Wie annotieren? Java Basics - Anfänger-Themen 1
O zweidimensionales array in eine csv-Datei Java Basics - Anfänger-Themen 1
CptK Richtigen Pfad beim einlesen von Datei finden Java Basics - Anfänger-Themen 2
E Pfadangaben bei Ausführbarer Jar Datei Java Basics - Anfänger-Themen 8
J Input/Output Konstruktor ergänzen, der zur Datei mit einem Objekt passt Java Basics - Anfänger-Themen 0
I Datei als Stream aus Ressource laden? Java Basics - Anfänger-Themen 2
LetsSebi Methode, die einen arry von objekten speichert in einer datei Java Basics - Anfänger-Themen 6
R Wie installiere ich Jdownloadersetup.sh datei mit debian Java Basics - Anfänger-Themen 2
J Csv-Datei einlesen Java Basics - Anfänger-Themen 52
H Daten aus einer Datei in eine Liste speichern Java Basics - Anfänger-Themen 23
W Java in Exe Datei umgewandelt, Ressourcen fehlen (Bilder und Audiodateien) Java Basics - Anfänger-Themen 1
A Fehler beim Ausführen einer class Datei Java Basics - Anfänger-Themen 6
L Konstruktor für eine Map-Datei/Map-Datei einlesen Java Basics - Anfänger-Themen 5
S CSV Datei auslesen und anders darstellen Java Basics - Anfänger-Themen 2
O zufälliges Wort aus einer Datei einlesen Java Basics - Anfänger-Themen 32
E Input/Output Datei anhand von Dateinamen einlesen Java Basics - Anfänger-Themen 5
C Datei über relative Pfade einlesen Java Basics - Anfänger-Themen 6
F Auto String mit Array Name aus Datei... oder so ähnlich Java Basics - Anfänger-Themen 4
O Best Practice Datei-Pfad zerlegen Java Basics - Anfänger-Themen 4
N Java CSV Datei auslesen Java Basics - Anfänger-Themen 6
J Ein Wort aus einer Datei zufällig ermitteln Java Basics - Anfänger-Themen 3
S Verbindung von einer Excel Datei zu Java-- Java Basics - Anfänger-Themen 4
G In Datei schreiben Java Basics - Anfänger-Themen 1
J Klassen .class Datei öffnen Java Basics - Anfänger-Themen 31
P Datei einlesen und zurückgeben Java Basics - Anfänger-Themen 5
G String wird nach Einlesen aus Datei nicht erkannt Java Basics - Anfänger-Themen 3
G Problem beim Speichern von Objekten in einer Datei Java Basics - Anfänger-Themen 7
J bundel - Datei wird nicht gefunden. Trotz STRG-SHIFT-J Java Basics - Anfänger-Themen 2
I ArrayList - Methode zum Speichern eines Eintrags in einer Datei Java Basics - Anfänger-Themen 17
C "HelloWorld" - Dateien erstellt, aber ist es eine class-Datei? Java Basics - Anfänger-Themen 2
A Klassen Datei als Kommandozeilenparameter übergeben Java Basics - Anfänger-Themen 8
J Datei auslesen (nur bestimmte Stellen) Java Basics - Anfänger-Themen 2
B Text-Datei nur in Stream speichern und dann ausgeben Java Basics - Anfänger-Themen 3
G Einbinden von Bildern in ausführbare Jar-Datei Java Basics - Anfänger-Themen 25
G If / While Programm (Datei auslesen) Java Basics - Anfänger-Themen 6
M Java: BufferedReader liest nur die Hälfte der Text-Datei. Java Basics - Anfänger-Themen 4
T PNG Datei wird nur manchmal geöffnet Java Basics - Anfänger-Themen 6

Ähnliche Java Themen

Neue Themen


Oben