C++ - Operatoren überladen
Transcription
C++ - Operatoren überladen
FB Informatik Prof. Dr. R.Nitsch C++ - Operatoren überladen Reiner Nitsch r.nitsch@fbi.h-da.de Überladen von Operatoren FB Informatik Prof. Dr. R.Nitsch • Ziel: Die üblichen Operatoren (z.B. +, -, /, *, %, (), [], <<, >>, <=, >=, =, ==, …) auch auf Objekte eigener Klassen anwenden können. Beispiel: Rechnen mit rationalen Zahlen Zähler Nenner void main() { Bruch b1(2,3),b2(3,4),b3; const Bruch kb(4,3); b3 = b1.mal(b2); geht, weil read-only b3 = kb.mal(b1); Methode b3 = b1.mal(kb); geht nur, weil const-Parameter //oder eleganter b3 = b1 * b2; äquivalente Anweisungen b3 = b1.operator*(b2); b3 = kb * b1; b3 = b1 * kb; geht, weil read-only Methode geht nur, weil const-Parameter } Principle of least privilege 19.10.2007 Bruch::Bruch(long Z,long N):z(Z),n(N) { } ~Bruch::Bruch() { } Bruch Bruch::mal(const Bruch & b) const { Bruch tmp(z*b.z,n*b.n); return tmp; //oder besser return Bruch(z*b.z,n*b.n); } lokales temporäres Objekt; wird auf Stack erzeugt, zurückgegeben, und danach Destruktoraufruf Bruch Bruch::operator*(const Bruch &b)const { return Bruch(z*b.z,n*b.n); } Farbcodierung class Bruch { erklären! long z,n; public: Bruch(long Z=0,long N=1); ~Bruch(); Bruch mal(const Bruch &b) const; Bruch operator*(const Bruch &b) const; }; C++ Operatoren überladen 2 Überladen von Operatoren FB Informatik Prof. Dr. R.Nitsch Aufgabe: Ergänzen Sie die Klasse Bruch so, dass die folgenden Anweisungen compilieren Keine Ergänzung notwendig: Bruch b4(5); Bruch(long=0, long=1) kann auch mit nur einem Parameter aufgerufen werden! b3 = b4 * 3; b3.operator*(const Bruch&) Typumwandlungskonstruktor aufrufen implizit Fortsetzung von main b3 = b4.operator*(3).operator*(b2); b3 = b4 * 3 * b2; // OK Bruch Welche Massnahme erlaubt es, den '*'-Operator hier mehrfach anzuwenden? Bruch b3 = 3 * b4 * b2; // Fehler entspricht b3 = 3.operator*(b4) long und nicht Bruch! Abhilfe: Globale operator* Funktion 19.10.2007 C++ Operatoren überladen 3 Globale operator-Funktion FB Informatik Prof. Dr. R.Nitsch Beispiel: Globale operator* Funktion Bruch& operator*( const int i, const Bruch &b) { return Bruch( i*b.z , n ); Zugriffsschutz, weil private // deshalb Bruch::operator* aufrufen return b*i; entspricht b.operator*(i) } jetzt geht auch b3 = 3 * b4 in main Warum? besser, weil universeller, ist aber die globale Funktion Bruch& operator*( const Bruch &a, const Bruch &b) { return Bruch( a.z*b.z , a.n*b.n ); } Problem 1: In Anweisung b3 = b4 * 3; kann Compiler nun nicht mehr entscheiden, ob ::operator* oder Bruch::operator* aufgerufen werden soll. Bruch::operator* löschen Problem 2: Zugriff auf private Attribute von aussen! public Attribute friend-Deklaration 19.10.2007 ☺ class Bruch { int z,n; Bruch(long Z=0,long N=1); ~Bruch(); Bruch& operator*(const Bruch &b) const; friend Bruch& operator*(const Bruch &a, const Bruch &b); }; C++ Operatoren überladen Anwendung in main ⇒ 4 Globale operator-Funktion FB Informatik Prof. Dr. R.Nitsch Anwendung in main b3 = 3 * b4; entspricht Aufruf operator*( Bruch(3), b4 ) b3 = b4 * 3; entspricht Aufruf operator*( b4 , Bruch(3) ) 19.10.2007 C++ Operatoren überladen 5 Was ist bei der Überladung von Operatoren für Klassen zu beachten? FB Informatik Prof. Dr. R.Nitsch • Es können keine neuen Operatoren erfunden werden • Linker Operand ist immer ein Objekt der Klasse • Die Operanden-Anzahl kann nicht geändert werden; unär bleibt unär, binär bleibt binär • Der Vorrang eines Operators bleibt erhalten • Die Operatoren . , :: , .* , ?: und sizeof() sind nicht überladbar • Soll der linke Operand kein Objekt einer Klasse sein, dann muß eine Globale Operatorfunktion definiert werden. aber: der rechte Operand muß dann ein Klassenobjekt sein. Grund: für Standardtypen können die Operatoren nicht umdefiniert werden. 19.10.2007 C++ Operatoren überladen 6 Globale Operatorfunktionen FB Informatik Prof. Dr. R.Nitsch • Einsatzgebiete: – Der Operator ist binär und der linke Operator kein Objekt einer Klasse (z.B. '+', '*',...) – Operator soll für fremde Klasse überladen werden ohne die Klasse selbst zu ändern, z.B. operator<< für Klasse ostream (kommt gleich) • Globale Operatorfunktionen sind nicht möglich für – die Operatoren '=', '()', '[]' und '->' 19.10.2007 C++ Operatoren überladen 7 Überladen des Operators << für ostream-Klasse FB Informatik Prof. Dr. R.Nitsch Zweck: Ausgeben von Objekten eigener Klassen, z.B. // cout ist Objekt der ostream-Klasse cout.operator<<(b1); // oder kompakter cout << b1; 1. Realisierungsmöglichkeit: die operator<< - Methode der ostream-Klasse überladen Syntax: // in <ostream> void ostream::operator<<(const Bruch &b){ (*this) << b.z << '/' << b.n; return; Problem? Zugriffschutz aufheben mittels friend-Deklaration! } Nachteil: 19.10.2007 Eingriff in eine System-Library notwendig! Finger weg C++ Operatoren überladen 8 Überladen des Operators << FB Informatik Prof. Dr. R.Nitsch 2. Realisierungsmöglichkeit: als operator<< -Methode der Klasse Bruch Syntax: void Bruch::operator<<( ostream& os ) { os << z << '/' << n; return; } Nachteil: ungewohnte Anwendung in Aufrufumgebung b1.operator<<(cout) entspricht b1 << cout; 3. Realisierungsmöglichkeit: Globale Funktion operator<< cout << b1; void operator<<( ostream &os, const Bruch &b ){ os << b.z << '/' << b.n; return; Problem? Zugriffschutz aufheben mittels friend-Deklaration! } 19.10.2007 C++ Operatoren überladen // Anwendung // Implementation 9 Ausgeben von mehreren Objekten in einer Anweisung FB Informatik Prof. Dr. R.Nitsch cout.operator<<(b1).operator<<('\n'); cout // oder kompakter cout << b1 << '\n'; cout Realisierungsmöglichkeit: cout cout << b1 << '\n'; ostream& void operator<<( ostream &os, const Bruch &b ){ os << b.z << '/' << b.n; return return;os; Zugriffschutz aufheben mittels friend-Deklaration! } 19.10.2007 C++ Operatoren überladen // Anwendung // Implementation 10 Globale Funktion operator>> für Klasse istream Zweck: Einlesen in eigene Klassen z.B. cin.operator>>(b3); cin >> b3; // oder kompakter cin ist Objekt der istream-Klasse FB Informatik Prof. Dr. R.Nitsch Trennzeichen sind - Leerzeichen ' ', - tab '\t', - CRLF-Zchn '\n' und - CR-Zchn '\r'. istream& operator>>( istream& is, Bruch &b) { // Implementation ohne Fehlerbehandlung // Zwischen Zähler- und Nenner-Einabe werden ausschliesslich Trennzeichen erwartet. char c; long Z, N; cout << "gib Bruch: Zaehler[Trennzeichen]Nenner "; is >> Z >> N; istream&::operator>>(int&) entfernt führende Trennzeichen; liest danach alle Ziffern bis zur 1. Nicht-Ziffer Fehlerflags (ios::failbit) werden bei unerwarteten Eingaben gesetzt, z.B. keine Ziffer(n) im Puffer, wenn solche erwartet werden. Klasse istream verwaltet folgende Statusbits vom Typ iostate ios::failbit: für einfache (behebbare) Fehler ios::badbit: für schwere (nicht behebbare) Fehler ios::eofbit: Das Ende des Bitstroms wurde erreicht ios::goodbit: alles ok 19.10.2007 Methoden zum Lesen der Statusbits bool istream::good() bool istream::fail() bool istream::bad() bool istream::eof() Methoden zum Setzen der Statusbits void iostream::setstate(iostate state) void iostream::clear() C++ Operatoren überladen 11 Probleme mit istream::operator(istream&, … ) Beispiel: Tastatureingabe 1/2 1 2 FB Informatik Prof. Dr. R.Nitsch 1.e2 2 int i1=0, i2=0; char c; cout << boolalpha; cin >> i1; cout << i1; cout << cin.good(); 1 true 1 true 1 true cin >> c; cout << c; cout << cin.good(); / true 2 true . true cin >> i2; cout << i2 cout << cin.good(); 2 true 0 false Nächstes Zeichen im Puffer ist 'e' 'e' ist kein int -> ios::fail wird gesetzt und operator>>(istream&,int) beendet i2 wurde nicht geändert -> i2 bleibt 0 Um mit cin wieder einlesen zu können muss das fail-Bit gelöscht werden: cin.fail(); Damit operator>>(istream&,int) wieder int-Eingaben verarbeitet muss zusätzlich - entweder das 'e' aus dem Lesepuffer entfernt werden: cin >> c; - oder der Eingabepuffer gelöscht werden: cin.sync(); 19.10.2007 C++ Operatoren überladen 12 Globale Funktion operator>> für Klasse istream FB Informatik Prof. Dr. R.Nitsch istream& operator>>( istream& is, Bruch &b) { // Implementation mit Fehlerbehandlung // Zwischen Zähler- und Nenner-Einabe werden ausschliesslich Trennzeichen erwartet. char c; long Z, N; cout << "gib Bruch: Zaehler[Trennzeichen]Nenner "; is >> Z >> N; if(!is.good()) return is; 1. Vorbedingung // Alternative dazu if(!( is >> Z >> N ).good()) return is; is // hier ggf. zusätzliche Vorbedingungen prüfen // z.B. nur pos. Brüche oder // nur echte Brüche // im Fehlerfall: is.setstate(ios::failbit) setzt das fail-Bit b.z = Z; Alle Vorbedingungen erfüllt! Jetzt wird Objektzustand verändert b.n = N; return is; 19.10.2007 C++ Operatoren überladen 13 Globale Funktion operator>> für Klasse istream FB Informatik Prof. Dr. R.Nitsch Ebenfalls gleichwertig, aber weniger selbstkommentierend ist if( is >> Z >> N ) { … } istream → bool Erklärung: Zur impliziten Typumwandlung von istream& nach bool enthält die Klasse istream eine Konvertierungsfunktion. Diese Funktion fragt mittels des Aufrufes this->good() die Statusbits des istream-Objekts ab und erzeugt den returnWert true nur dann, wenn kein Fehlerbit gesetzt ist. Das eigentliche Einlesen ist hier ein Seiteneffekt. 19.10.2007 C++ Operatoren überladen 14 Anwendung der globalen Funktion operator>> für Klasse istream FB Informatik Prof. Dr. R.Nitsch void main() { Bruch b1(2,3), b2(3,4), b3; const Bruch kb(4,3); b3 = b1 * b2; Die Bedingung !(cin >> b3) ruft operator!() der istreamKlasse auf, der den Wert true liefert, wenn ein Fehlerbit gesetzt ist. In diesem Fall müssen die Fehlerflags gelöscht werden und der while( !(cin >> b3) ) resetcin(); Eingabepuffer geleert werden. Dieser Aufgabe übernimmt die userdefinierte globale Funktion resetcin() // oder Operatoren-Hierarchie . (oben) while( !(cin >> b3 ).good()) resetcin(); ! } >> (unten) void resetcin() { cin.clear(); cin.sync(); } 19.10.2007 löscht die Fehlerflags leert den Eingabe-Puffer C++ Operatoren überladen 15 Fall-Studie: Dynamisches Array FB Informatik Prof. Dr. R.Nitsch • Arrays in C++ – Keine Indexprüfung – Vergleich von Arrays mit == nicht definiert – Zuweisung nicht definiert (Array-Namen sind const Zeiger) – können nur elementweise ein- und ausgegeben werden. • Beispiel: Implementierung einer (dynamischen) Arrayklasse mit – Copy-Konstruktor – Indexoperator operator[], der auch den Index prüft – Kenntnis der Anzahl gespeicherter Elemente – Zuweisungs operator= – Eingabe/Ausgabe des gesamten Arrays mit operator<< und operator>> – Array-Vergleichsoperatoren operator== und operator!= 19.10.2007 C++ Operatoren überladen 16 Fall-Studie: Dynamische Array-Klasse DynArray FB Informatik Prof. Dr. R.Nitsch // DynArray für Elemente vom Typ 'float' class DynArray { Anzahl der Elemente public: defaultFunktion constructor class Array Welche hat für diese Komponente? DynArray( int arraySize=0, float value=0.0 ); // Array Welche Funktionfür hatclass diese Komponente? DynArray( const DynArray& toCopy ); // copy-constructor // destructor für class Array ~DynArray(); int getSize() const { return size; } // Überladener Zuweisungsoperator; DynArray& operator= ( const DynArray& toAssign ); // return-Wert erlaubt a1= a2 = a3 float& operator[] ( int i ); private: int size; float* pArr; // Überladener Index-Operator für nicht-const Arrays // return "by reference" erzeugt einen lvalue (read&write) Größe des Arrays Zeiger auf ein C-Array mit Elementen vom Typ float }; // DynArray 19.10.2007 C++ Operatoren überladen 17 Testanwendung DynArray void main() { DynArray float1(3); DynArray float2(0); FB Informatik Size of array float1 is 3 Prof. Dr. R.Nitsch Array after initialization: 0 0 0 // 3-element Array // 0-element Array by default // Größe cout << << << und Inhalt von float1 ausgeben "Size of array float1 is " float1.getSize() "\nArray after initialization:\n" << float1; // Größe cout << << << und Inhalt von float2 ausgeben "\nSize of array float2 is " float2.getSize() "\nArray after initialization:\n" << float2; // float1 und float2 einlesen cout << "\nInput 3 float:\n"; for( int i=0; i<3; i++) cin >> float1[i]; cout << "\nInput 5 float:\n"; for( i=0; i<5; ++i) cin >> float2[i]; Size of array float1 is 0 Array after initialization: Input 3 float: 1.1 2.2 3.3 Input 5 float: 4.4 5.5 6.6 7.7 8.8 After input, the arrays contain: float1: 1.1 2.2 3.3 float2: 4.4 5.5 6.6 7.7 8.8 // float1 und float2 ausgeben cout << "\nAfter input, the arrays contain:\n" << "float1:\n" << float1 << "float2:\n" << float2; 19.10.2007 C++ Operatoren überladen 18 Auswerten: float1 != float2 float1 und float2 sind nicht gleich Testanwendung DynArray FB Informatik Prof. Dr. R.Nitsch float3 hat die Groesse 3 Arraywerte nach der Initialisierung: 1.1 2.2 3.3 // Testen des überladenen Ungleich-Operators (!=) cout << "\nAuswerten: float1 != float2\n"; if ( float1 != float2 ) cout << "float1 und float2 sind nicht gleich\n"; Zuweisung float2 = float1 : float1: // Test des Copy-Konstruktors 4.4 5.5 6.6 7.7 // array float3 erzeugen und mit Werten 8.8 // aus float1 initialisieren float2: DynArray float3( float1 ); 4.4 5.5 6.6 7.7 8.8 // Größe und Inhalt von float3 ausgeben cout << "\nfloat3 hat die Groesse " << float3.getSize() << "\nArraywerte nach der Initialisierung:\n" << float3; // Testen des überladenen Zuweisungsoperator (=) cout << "\nZuweisung float2 = float1 :\n"; float1 = float2; // beachte, dass float1 nur aus 3 Elementen besteht. cout << "float1:\n" << float1 << "float2:\n" << float2; 19.10.2007 C++ Operatoren überladen 19 Implementation der Array-Klasse DynArray FB Informatik Prof. Dr. R.Nitsch #include "DynArray.h" // default Konstruktor für Klasse Array DynArray::DynArray( int arraySize, float value ) { if( arraySize > 0) // Wenn Array nicht leer sein soll { size = arraySize; pArr = new float[arraySize]; // Speicherplatz vom System anfordern for( int i=0; i< arraySize; i++ ) pArr[i]=value; // elementweise mit value initialisieren } else { size = 0; pArr = NULL; } } // Ende Array default Konstruktor // Destruktor für Klasse Array DynArray::~DynArray() { if (pArr!=NULL) delete [] pArr; } 19.10.2007 C++ Operatoren überladen // Speicherplatz freigeben 20 Implementation der Array-Klasse DynArray FB Informatik Prof. Dr. R.Nitsch Warum genügt hier nicht die default-Zuweisung, die elementweise kopiert? DynArray A1(3), A2(2); A1 A2[1] = 1; A1 = A2; A2 size=3 pArr size=2 pArr Heap verwitwetes Objekt Heap 0 0 size=2 0 0 A1 pArr 0 0 0 1 A2 size=2 pArr 0 1 A1 size=2 pArr Heap 0 1 A2 size=2 pArr 0 1 "flache" Zuweisung (default) "tiefe" Zuweisung. // Zuweisungsoperator // Rückgabewert ermöglicht: a1 = a2 = a3 DynArray& DynArray::operator= (const DynArray& toAssign) { if ( &toAssign != this ) { // check for self-assignment // Bei verschiedenen Array-Größen, linkes Orginal löschen, // dann Speicherplatz für neues linksseitiges Array anfordern if ( size>0 && size != toAssign.size ) { delete [] pArr; // bisherigen Speicherplatz freigeben size = toAssign. size; // Größe des neuen Objekts festlegen pArr = new float[ size ]; // Speicherplatz gemäß neuer Größe anfordern } for ( int i=0; i<size; i++ ) pArr[i] = toAssign.pArr[i]; // Array elementweise kopieren } return *this; } 19.10.2007 C++ Operatoren überladen 21 Implementation der Array-Klasse DynArray FB Informatik Prof. Dr. R.Nitsch // Copy-Konstruktor für class DynArray; // call "by reference", um nicht abbrechenden rekursiven Aufruf zu vermeiden DynArray::DynArray( const DynArray& arrayToCopy ) { size = arrayToCopy.size; pArr = new float[ size ]; // Speicherplatz für neues Array anfordern for ( int i = 0; i < size; i++ ) // Arraywerte kopieren pArr[ i ] = arrayToCopy.pArr[ i ]; } // end DynArray copy constructor // globale operator<< Funktion für class DynArray ostream& operator<<( ostream& os, const DynArray& a ) { int i; // Ausgabe aller Arrayelemente for ( i = 0; i < a.size; i++ ) { os << setw( 12 ) << a.pArr[ i ]; if ( ( i + 1 ) % 4 == 0 ) // 4 Werte pro Ausgabezeile os << endl; } // end for if ( i % 4 != 0 ) // Zeilenvorschub am Zeilenende os << endl; return os; // ermöglicht cout << x << y; } // end function operator<< 19.10.2007 C++ Operatoren überladen 22 Implementation der Array-Klasse DynArray FB Informatik Prof. Dr. R.Nitsch // Überladener Index-Operator für r/w-Arrays // gibt r/w-Referenz auf gewünschtes Element zurück // Keine Index-Bereichskontrolle float& DynArray::operator[] ( int i ) { return *(pArr + i); // Referenz auf Element mit Index i zurückgeben // oder alternativ return pArr[i]; } // Überladener operator!= bool DynArray::operator!= ( const DynArray& other ) const { if( size != other.size ) return true; for( int i=0; i<size; ++i ) { if( pArr[i] != other.pArr[i] ) return true; } return false; } // Überladener operator!= bool DynArray::operator== ( const DynArray& other ) const { return *this != other; } 19.10.2007 C++ Operatoren überladen 23 Testanwendung DynArray void main() { DynArray float1(3); DynArray float2(0); FB Informatik Prof. Dr. R.Nitsch Size of array float1 is 3 Array after initialization: // 3-element Array // 0-element Array by default // Größe cout << << << und Inhalt von float1 ausgeben "Size of array float1 is " float1.getSize() "\nArray after initialization:\n" << float1; // Größe cout << << << und Inhalt von float1 ausgeben "\nSize of array float2 is " float2.getSize() "\nArray after initialization:\n" << float2; // float1 und float2 einlesen cout << "\nInput 3 float-values:\n"; for( int i=0; i<3; i++) cin >> float1[i]; cout << "\nInput 5 float:\n"; for( int i=0; i<5; ++i) cin >> float2[i]; 0 0 0 Size of array float2 is 0 Array after initialization: Input 3 float-values: 1.1 2.2 3.3 Input 5 float-values: 4.4 5.5 6.6 7.7 8.8 After input, the arrays contain: float1: 1.1 float2: 4.4 8.8 2.2 3.3 5.5 6.6 7.7 // float1 und float2 ausgeben cout << "\nAfter input, the arrays contain:\n" << "float1:\n" << float1 << "float2:\n" << float2; 19.10.2007 C++ Operatoren überladen 25 Auswerten: float1 != float2 float1 und float2 sind nicht gleich Testanwendung DynArray FB Informatik Prof. Dr. R.Nitsch float3 hat die Groesse 3 // Testen des überladenen Ungleich-Operators (!=) Arraywerte nach der Initialisierung: cout << "\nAuswerten: float1 != float2\n"; 1.1 2.2 3.3 if ( float1 != float2 ) cout << "float1 und float2 sind nicht gleich\n"; Zuweisung float2 = float1 : float1: // Test des Copy-Konstruktors 4.4 5.5 6.6 7.7 // array float3 erzeugen und mit Werten 8.8 // aus float1 initialisieren float2: DynArray float3( float1 ); 4.4 5.5 6.6 7.7 8.8 // Größe und Inhalt von float3 ausgeben cout << "\nfloat3 hat die Groesse " << float3.getSize() << "\nArraywerte nach der Initialisierung:\n" << float3; // Testen des überladenen Zuweisungsoperator (=) cout << "\nZuweisung float2 = float1 :\n"; float1 = float2; // beachte, dass float1 nur aus 3 Elementen besteht. cout << "float1:\n" << float1 << "float2:\n" << float2; 19.10.2007 C++ Operatoren überladen 26 Testanwendung DynArray FB Informatik Prof. Dr. R.Nitsch // Testen des überladenen Gleichheitsoperators (==) cout << "\nAuswerten: float1 == float2\n"; if ( float1 == float2 ) cout << "float1 und float2 sind gleich\n"; // Testen des überladenen Index-Operators ([]) cout << "\nfloat1[4] = 1000;\n"; float1[4] = 1000; // schreibender Zugriff (rvalue) cout << "float1:\n" << float1; // lesender Zugriff (lvalue) } // end main Auswerten: float1 == float2 float1 und float2 sind gleich float1[5]=1000; float1: 4.4 5.5 6.6 1000 7.7 Press any key to continue 19.10.2007 C++ Operatoren überladen 27