Tematyka i cel ćwiczenia.
Celem ćwiczenia jest zapoznanie z definiowaniem funkcji operatorowych jako funkcji składowych klasy lub funkcji zaprzyjaźnionych oraz ze standardowymi strumieniami wejścia i wyjścia cin i cout.
Wymienione wyżej pojęcia są zilustrowane na przykładzie klasy implementującej strukturę danych typu stos wykorzystywanej w poprzednim ćwiczeniu.
Wprowadzenie.
W języku C++ można definiować funkcje określające znaczenie następujących operatorów:
+ - * / % ^ & | ~ !
= < > += -= *= /= %= ^= &=
|= << >> >>= <<= == != <= >= &&
|| ++ -- ->* , -> [] () new delete
Definiując nowe znaczenie wyżej wymienionych operatorów nie można zmieniać składni ani priorytetów tych operatorów. Oznacza to, że nie można przykładowo zdefiniować jednoargumentowego operatora, który standardowo jest dwuargumentowy (np. %) lub odwrotnie (np. !). Nie można też definiować nowych symboli operatorów.
Funkcja operatorowa musi być składową klasy albo mieć co najmniej jeden argument będący klasą lub typem wyliczeniowym (enum). Oznacza to, że definiowane funkcje operatorowe mogą operować na typach zdefiniowanych przez użytkownika. Nie można więc zmieniać standardowego znaczenia operatorów (np. normalnego znaczenia operatorów +, - dla typu int).
Jeśli przykładowo zadeklaruje się następującą klasę:
class M {
// składowe klasy
public :
M operator + (M am);
};
to w programie można użyć następujących konstrukcji:
M m1, m2; // deklaracja obiektów klasy M
m1+m2; // wywołanie funkcji operatorowej operator +
W chwili napotkania wyrażenia m1+m2 kompilator wywoła zdefiniowaną w klasie M funkcję operatorową operator + w następujący sposób:
m1.operator+(m2);
Należy pamiętać, że kompilator nie czyni żadnych założeń co do dodatkowych własności operatorów. Nie można więc założyć, że zdefiniowany przez użytkownika operator + albo * będzie przemienny (tak jak dodawanie czy mnożenie liczb).
Jako przykład niech posłuży definicja funkcji operator *, którego jednym argumentem będzie obiekt typu class M, a drugim liczba całkowita typu int.
Jeżeli funkcję operator* zdefiniować jako funkcję składową klasy (tak jak funkcję operator+ w przykładzie powyżej):
class M {
// składowe klasy
public :
M operator * (int n);
};
M M::operator * (int n)
{
// definicja funkcji operator *
}
to po odpowiednich deklaracjach, w momencie napotkania wyrażenia (***):
M m1; //deklaracja obiektu klasy M
int k = 3;
m1*k; // wyrażenie (***)
zostanie wywołana funkcja operatorowa m1.operator*(int). Gdyby w programie znalazło się wyrażenia:
k*m1;
to wystąpiłby błąd, ponieważ nie ma możliwości zdefiniowana funkcja operatorowej int.operator*(M) jako funkcji składowej klasy. Problem ten można rozwiązać deklarując klasę M z dwoma operatorowymi funkcjami zaprzyjaźnionymi:
class M {
// składowe klasy
public :
friend M operator * (M m, int n);
friend M operator * (int n, M m);
};
M operator * (M m, int n)
{
// definicja funkcji operator *
}
M operator * (int n, M m)
{
// definicja funkcji operator *
}
W takim przypadku kompilator wybierze jedną z funkcji w zależności o typów argumentów po lewej i prawej stronie operatora *.
Do ćwiczenia należy zapoznać się z literatury z następującymi pojęciami:
funkcje operatorowe i ich definiowanie,
standardowe strumienie wejścia/wyjścia w języku C++: cin oraz cout.
Z literatury można polecić następujące książki:
B. Stroustrup „Język C++”,
S. Lipmann „Podstawy języka C++”.
Program ćwiczenia.
Na bazie klasy typu Stack zdefiniowanej w poprzednim ćwiczeniu:
Zastąpić metody push(), pop(), empty() i full() wewnętrznymi metodami operatorowymi. Deklarację tych metod mogą być następujące:
Stack & operator << (int Ax); // push()
Stack & operator >> (int &Ax); // pop()
int operator ! (void); // empty()
int operator ~ (void); // full()
Napisać definicje wyżej zadeklarowanych metod operatorowych oraz zastąpić w funkcji main() wywołania metod push(), pop(), empty() i full() odpowiednimi metodami operatorowymi. Uruchomić tę część ćwiczenia jako program nr 1.
Zastąpić zdefiniowane w punkcie 1 metody operatorowe funkcjami operatorowymi zdefiniowanymi jako funkcje zewnętrzne zaprzyjaźnione z klasą. Deklaracje tych metod mogą być następujące:
friend Stack & operator << (Stack &As, int Ax);
friend Stack & operator >> (Stack &As, int &Ax);
friend int operator ! (Stack As);
friend int operator ~ (Stack As);
Napisać definicje wyżej zadeklarowanych funkcji. Uruchomić tę część ćwiczenia jako program nr 2.
W funkcji main() operacje wejścia/wyjścia programu, które są zaimplementowane standardowymi funkcjami języka C scanf() i printf() zastąpić mechanizmami dostępnymi w języku C++: strumieniami cin oraz cout oraz manipulatorami formatującymi. Nie zapomnieć o konieczności dołączenia plików nagłówkowych iostream.h i iomanip.h.
JĘZYK C++, FUNKCJE OPERATOROWE 1
Katedra Automatyki Napędu i Urządzeń Przemysłowych AGH