Präsentation herunterladen
Die Präsentation wird geladen. Bitte warten
Veröffentlicht von:Adelhard Durk Geändert vor über 11 Jahren
2
EINI-I Einführung in die Informatik für Naturwissenschaftler und Ingenieure I Kapitel 9 Claudio Moraga; Gisbert Dittrich FBI Unido moraga@cs.uni-dortmund.de
3
Kap 9: Durchlaufstrategien Vorl EINI-I" 2 09.01.2001 Gliederung Kapitel 9 Einfache Dateibehandlung Beispiel: Wörter zählen –Problem, Datenstruktur –Einfügen: Strategie + Implementierung –Alphabetisch geordnete Ausgabe: Strategie + Impl. Durchlaufstrategien bei Bäumen –In die Tiefe Inorder Präorder Postorder –In die Breite
4
Kap 9: Durchlaufstrategien Vorl EINI-I" 3 09.01.2001 Dateien Einfache Dateibehandlung –(externe Dateien, Öffnen, Lesen/Schreiben, Schließen). –Am Beispiel: Problem, die Wörter in einem gegebenen Text zu zählen und sie mit ihrer Häufigkeit alphabetisch geordnet auszugeben. Zunächst: Dateien Lies aus einer Eingabe-Datei, schreibe in eine Ausgabe-Datei. Am elementaren Beispiel. Programm
5
Kap 9: Durchlaufstrategien Vorl EINI-I" 4 09.01.2001 Dateien #include bindet die Bibliothek zur Dateibehandlung ein Bindung von Datei-Variablen (im Programm) an Dateien (im Dateisystem) beim Öffnen der Datei. ifstream *Eingabedatei; EingabeDatei = new ifstream(EinDat); Benutzung von Dateien: wie Standard Ein-/Ausgabe –ifstream : Eingabe - ofstream : Ausgabe Benutzung: wie cin (bzw. cout ) –*Eingabedatei >> gelesen;
6
Kap 9: Durchlaufstrategien Vorl EINI-I" 5 09.01.2001 Dateien Testen: (*Eingabedatei).eof() (liefert "true" genau dann, wenn das Ende der Datei erreicht ist; eof: end of file) !(*Eingabedatei).eof() (liefert "true" genau dann, wenn das Ende der Datei noch nicht erreicht ist) Analog: ofstream *AusgabeDatei; Werden Dateien zum Schreiben geöffnet, so geht meist ihr vorheriger Inhalt verloren.
7
Kap 9: Durchlaufstrategien Vorl EINI-I" 6 09.01.2001 // K9-P1 // Elementare Verwendung von Dateien // Vorsicht: nicht robust #include const int maxLen = 70; void Einlesen(ifstream *, ofstream *); void Schreiben(char *, ofstream *); 1
8
Kap 9: Durchlaufstrategien Vorl EINI-I" 7 09.01.2001 main() { ifstream *EingabeDatei; ofstream *AusgabeDatei; char EinDat[80], AusDat[80]; cout << "bitte Eingabedatei angeben: "; cin >> EinDat; cout << "bitte AusgabeDatei angeben: "; cin >> AusDat; cout << "Danke" << endl; EingabeDatei = new ifstream(EinDat); AusgabeDatei = new ofstream(AusDat); Einlesen(EingabeDatei, AusgabeDatei); cout << "fertig" << endl; } 2
9
Kap 9: Durchlaufstrategien Vorl EINI-I" 8 09.01.2001 void Einlesen(ifstream *ein, ofstream *aus) { char gelesen[maxLen]; *ein >> gelesen; while (!(*ein).eof()) { cout << "gelesen: " << gelesen << "\t\t\t\t"; Schreiben (gelesen, aus); *ein >> gelesen; } void Schreiben(char *Wort, ofstream *aus) { if (Wort != NULL) { *aus << Wort << endl;// Wort mit..... //..Zeilenumbruch in die Datei schreiben. cout << "geschrieben: " << Wort << endl; // Wort mit Zeilenumbruch... //..auf den Bildschirm schreiben. }}//3 Ausführen
10
Kap 9: Durchlaufstrategien Vorl EINI-I" 9 09.01.2001 Wörter zählen Problem: –zähle die Wörter in einem gegebenen Text, –gib sie mit ihrer Häufigkeit alphabetisch geordnet aus. Datenstruktur: –binärer Suchbaum (--> Wörter lassen sich ordnen) –erweitert um einen Zähler.
11
Kap 9: Durchlaufstrategien Vorl EINI-I" 10 09.01.2001 Datenstruktur zum Zählen von Wörtern text zaehler LSohnRSohn struct BinBaum { char text[maxLen]; int zaehler; BinBaum * LSohn, *RSohn; } ;
12
Kap 9: Durchlaufstrategien Vorl EINI-I" 11 09.01.2001 Strategie zum Einfügen Suchen nach einer Zeichenkette im binären Suchbaum –Zeichenkette nicht gefunden: neuen Knoten einfügen, Zähler zu 1 initialisieren –Zeichenkette gefunden: Zähler um 1 erhöhen
13
Kap 9: Durchlaufstrategien Vorl EINI-I" 12 09.01.2001 void strcpy(char *, char *); int strcmp(char *, char *); BinBaum * Einfuegen(BinBaum *B, char * k) { if (B == NULL) { BinBaum *Hilf = new BinBaum; strcpy(Hilf->text, k); Hilf->zaehler = 1; Hilf->LSohn = Hilf->RSohn = NULL; B = Hilf; Hilf = NULL; } else { int Vergl = strcmp(B->text,k); if (Vergl < 0) B->RSohn = Einfuegen(B->RSohn, k); else if (Vergl > 0) B->LSohn = Einfuegen(B->LSohn, k); else if (Vergl == 0) B->zaehler += 1; } return B; } Text noch nicht gesehen Text bekannt: einfügen, Zähler erhöhen
14
Kap 9: Durchlaufstrategien Vorl EINI-I" 13 09.01.2001 Alphabetisch geordnete Ausgabe Behauptung: Nachfolgender Durchlauf liefert als Resultat: –geordnete Ausgabe Durchlaufstrategie: –Durchlaufe den binären Suchbaum mit Wurzel w rekursiv wie folgt: Durchlauf durch den linken Unterbaum von w Ausdruck (der Nutzinfo) der Wurzel w Durchlauf durch den rechten Unterbaum von w
15
Kap 9: Durchlaufstrategien Vorl EINI-I" 14 09.01.2001 Beispiel (Baumdurchlauf) 47 6 17 18 23 26 46717182326 47 17 18 23 26 6
16
Kap 9: Durchlaufstrategien Vorl EINI-I" 15 09.01.2001 Beweis Durch vollständige Induktion nach der Anzahl der Knoten –Der Induktionsbeginn (kein Knoten) ist erfüllt –Der Induktionsschritt: der linke Unterbaum wird geordnet ausgegeben (IV), dann wird die Wurzel ausgegeben, dann wird der rechte Unterbaum geordnet ausgegeben (IV). (Die Wurzel steht bzgl. der Ordnung "in der Mitte".)
17
Kap 9: Durchlaufstrategien Vorl EINI-I" 16 09.01.2001 Programmtext: Ausdrucken void KnotenDruck(BinBaum *, ofstream *); void Ausdrucken(BinBaum *K, ofstream *aus) { if (K != NULL) { Ausdrucken(K->LSohn, aus); KnotenDruck(K, aus); Ausdrucken(K->RSohn, aus); } } Programm
18
Kap 9: Durchlaufstrategien Vorl EINI-I" 17 09.01.2001 Anmerkungen Zugriff auf Ein- und Ausgabedateien wird über Zeiger auf ifstream- und ofstream- Variablen bewirkt. Initialisierungen –EingabeDatei = new ifstream(EinDat); –AusgabeDatei = new ofstream(AusDat); Varianten (Ausgabedateien als Konstante bekannt) z.B.: –ofstream *Ausgabe = new ofstream("von.aus");
19
Kap 9: Durchlaufstrategien Vorl EINI-I" 18 09.01.2001 BinBaum * Einfuegen(BinBaum *, char *); BinBaum * Einlesen(ifstream *ein) { BinBaum *bst = NULL; char gelesen[maxLen]; *ein >> gelesen; while (!(*ein).eof()) { bst = Einfuegen(bst, gelesen); *ein >> gelesen; } return bst; } Feinheiten: *ein und (*ein).eof()
20
Kap 9: Durchlaufstrategien Vorl EINI-I" 19 09.01.2001 Analog: Ausgabe void Schreiben(char *, int, ofstream *); void KnotenDruck(BinBaum *T, ofstream *aus){ Schreiben(T->text, T->zaehler, aus); } void Schreiben( char * s, int k, ofstream *aus ){ *aus << k << "\t\t\t" << s << endl; }
21
Kap 9: Durchlaufstrategien Vorl EINI-I" 20 09.01.2001 // K9-P2 Binärer Suchbaum // #include // beinhaltet void strcpy(char *, char *); // und int strcmp(char *, char *); const int maxLen = 70; struct BinBaum { char text[maxLen]; int zaehler; BinBaum * LSohn, *RSohn; }; // 1 Bibliothek für die Dateibehandlung
22
Kap 9: Durchlaufstrategien Vorl EINI-I" 21 09.01.2001 // Funktionsprototypen BinBaum * Einlesen(ifstream *); BinBaum * Einfuegen(BinBaum *, char *); void Ausdrucken(BinBaum *, ofstream *); void KnotenDruck(BinBaum *, ofstream *); void Schreiben(char *, int, ofstream *); // 2
23
Kap 9: Durchlaufstrategien Vorl EINI-I" 22 09.01.2001 main() { BinBaum *BST; ifstream *EingabeDatei; ofstream *AusgabeDatei; char EinDat[80], AusDat[80]; cout << "bitte Eingabedatei angeben: "; cin >> EinDat; cout << "bitte AusgabeDatei angeben: "; cin >> AusDat; cout << "Danke" << endl; EingabeDatei = new ifstream(EinDat); AusgabeDatei = new ofstream(AusDat); BST = Einlesen(EingabeDatei); Ausdrucken(BST, AusgabeDatei); cout << "fertig" << endl; } // 3
24
Kap 9: Durchlaufstrategien Vorl EINI-I" 23 09.01.2001 BinBaum * Einlesen(ifstream *ein) { BinBaum *bst = NULL; char gelesen[maxLen]; *ein >> gelesen; while (!(*ein).eof()) { bst = Einfuegen(bst, gelesen); *ein >> gelesen; }; return bst; } // 4
25
Kap 9: Durchlaufstrategien Vorl EINI-I" 24 09.01.2001 BinBaum * Einfuegen(BinBaum *B, char * k) { if (B == NULL) { BinBaum *Hilf = new BinBaum; strcpy(Hilf->text, k); Hilf->zaehler = 1; Hilf->LSohn = Hilf->RSohn = NULL; B = Hilf; Hilf = NULL;} else { int Vergl = strcmp(B->text,k); if (Vergl < 0) B->RSohn = Einfuegen(B->RSohn, k); else if (Vergl > 0) B->LSohn = Einfuegen(B->LSohn, k); else if (Vergl == 0) B->zaehler += 1;} return B;} // 5
26
Kap 9: Durchlaufstrategien Vorl EINI-I" 25 09.01.2001 void Ausdrucken(BinBaum *K, ofstream *aus) { if (K != NULL) { Ausdrucken(K->LSohn, aus); KnotenDruck(K, aus); Ausdrucken(K->RSohn, aus); } void KnotenDruck(BinBaum *T, ofstream *aus){ Schreiben(T->text, T->zaehler, aus); } void Schreiben(char * s, int k, ofstream *aus){ *aus << k << "\t\t\t" << s << endl; } // 6
27
Kap 9: Durchlaufstrategien Vorl EINI-I" 26 09.01.2001 Durchlauf durch Bäume 47 6 17 18 23 26 Diese Art des Durchlaufs heißt Inorder-Durchlauf. w BLinksBRechts Inorder () Inorder(Wurzel BLinks) Druck(w) Inorder(Wurzel BRechts) = 46717182326
28
Kap 9: Durchlaufstrategien Vorl EINI-I" 27 09.01.2001 w BLinksBRechts Präorder ( ) Präorder(Wurzel BLinks) Druck(w) Präorder(Wurzel BRechts) = 17647231826 17 6 47 23 1826
29
Kap 9: Durchlaufstrategien Vorl EINI-I" 28 09.01.2001 w BLinks BRechts Postorder ( ) Postorder(Wurzel BLinks) Druck(w) Postorder(Wurzel BRechts) = 47 6 1826 23 17 47618262317
30
Kap 9: Durchlaufstrategien Vorl EINI-I" 29 09.01.2001 Durchlaufstrategien Strategie bei allen drei Durchlaufarten heißt Tiefensuche: Es wird zunächst in die Tiefe und nicht in die Breite gegangen. Alternative: Breitensuche Trage den Baum "schichtenweise" ab.
31
Kap 9: Durchlaufstrategien Vorl EINI-I" 30 09.01.2001 Beispiel Breitensuche 623471826 17 623471826
32
Kap 9: Durchlaufstrategien Vorl EINI-I" 31 09.01.2001 Idee zur Implementation Verwalte die Knoten in einer "Warteschlange" ein Knoten wird gedruckt seine Söhne werden in die Warteschlange eingefügt bis die Warteschlange leer ist Initialisierung der Warteschlange mit der Wurzel des Baums.
33
Kap 9: Durchlaufstrategien Vorl EINI-I" 32 09.01.2001 Beispiel 17623471826 17 623 471826
Ähnliche Präsentationen
© 2025 SlidePlayer.org Inc.
All rights reserved.