- •2.5. Простейший ввод и вывод
- •2.5.1. Объект cin
- •2.5.2. Объект cout
- •2.5.3. Манипуляторы
- •2.6. Операторы для динамического выделения и освобождения памяти (new и delete)
- •5. Перегрузка
- •5.1. Перегрузка функций
- •5.2. Перегрузка операторов
- •5.2.1. Перегрузка бинарного оператора
- •5.2.2. Перегрузка унарного оператора
- •5.2.3. Дружественная функция operator
- •5.2.5. Перегрузка оператора []
- •5.2.6. Перегрузка оператора ()
- •5.2.8. Перегрузка операторов new и delete
- •6. Шаблоны
- •6.1. Параметризированные классы
- •6.2. Передача в шаблон класса дополнительных параметров
- •6.3. Шаблоны функций
- •6.4. Совместное использование шаблонов и наследования
- •6.5. Шаблоны класса и friend-функции
- •7.1. Организация ввода-вывода
- •7.2. Состояние потока
- •7.3. Строковые потоки
- •7.4. Организация работы с файлами
- •7.5. Организация файла последовательного доступа
- •Istream& seekg( streampos pos );
- •Istream& seekg( streamoff off, ios::seek_dir dir );
- •7.6. Создание файла произвольного доступа
- •7.7. Основные функции классов ios, istream, ostream
- •8.1. Основы обработки исключительных ситуаций
- •8.2. Перенаправление исключительных ситуаций
- •8.3. Исключительная ситуация, генерируемая оператором new
- •8.4 Генерация исключений в конструкторах
- •8.5. Задание собственной функции завершения
- •8.6. Спецификации исключительных ситуаций
- •8.7. Задание собственного неожиданного обработчика
- •8.8. Иерархия исключений стандартной библиотеки
5.2.1. Перегрузка бинарного оператора
Функция operator для перегрузки (доопределения) бинарных операторов может быть описана двумя способами:
как компонента-функция класса с одним аргументом;
как глобальная функция (функция, описанная вне класса) с двумя аргументами.
При перегрузке бинарного оператора # выражение a#b может быть представлено при первом способе как a.operator#(b) или как operator #(a,b) при втором способе перегрузки.
Рассмотрим простой пример переопределения операторов *, = , > и == по отношению к объекту, содержащему декартовы координаты точки на плоскости. В примере использован первый способ перегрузки.
#include <iostream>
using namespace std;
class dek_koord
{ int x,y; // декартовы координаты точки
public:
dek_koord(){};
dek_koord(int X,int Y): x(X),y(Y) {}
dek_koord operator*(const dek_koord);
dek_koord operator=(const dek_koord);
dek_koord operator>(const dek_koord);
int operator ==(const dek_koord);
void see();
};
dek_koord dek_koord::operator*(dek_koord a) // перегрузка операции *
{ dek_koord tmp; // локальный объект
tmp.x=x*a.x;
tmp.y= y*a.y;
return tmp;
}
dek_koord dek_koord::operator =(const dek_koord a)
{ x=a.x; // перегрузка операции =
y=a.y;
return *this;
}
dek_koord dek_koord::operator >(const dek_koord a)
{ if (x<a.x) x=a.x; // перегрузка операции >
if (y<a.y) y=a.y;
return *this;
}
int dek_koord::operator ==(const dek_koord a) // перегрузка операции ==
{ if (x==a.x && y==a.y) return 0; // 0 – координаты равны
if (x>a.x && y>a.y) return 1; //
if (x<a.x && y<a.y) return -1; //
else return 2; // неопределенность
}
void dek_koord::see() // функция просмотра содержимого объекта
{ cout << "координата х = " << x << endl;
cout << "координата y = " << y << endl;
}
int main()
{ dek_koord A(1,2), B(3,4), C;
int i;
A.see();
B.see();
C=A*B; // вначале перегрузка операции * затем =
C.see();
C=A>B; // компоненты объекта С принимают значение max от А и В
C.see();
i=A==B; // i получает значение сравнения A==B (−1,0,1,2….)
// cout << A==B << endl; // ошибка
// error binary '<<' : no operator defined which takes a right-hand operand
// of type 'class dek_koord' (or there is no acceptable conversion)
cout << (A==B) << endl; // верно
}
Результат работы программы:
координата х = 1
координата y = 2
координата х = 3
координата y = 4
координата х = 3
координата y = 8
координата х = 3
координата y = 4
В приведенной выше программе функцию перегрузки оператора * можно изменить, например, следующим образом:
dek_koord &dek_koord::operator*(const dek_koord &a)
{ x*=a.x;
y*=a.y;
return *this;
}
В этом примере функция operator* в качестве параметра получает ссылку на объект, стоящий в правой части выражения А*В, т.е. на В. Ссылка – это второе имя (псевдоним) для одного и того же объекта. Более подробно ссылки будут рассмотрены ниже. Функция operator* при вызове получает скрытый указатель на объект А и модифицирует неявные параметры (компоненты-данные объекта А – х и у). Возвращается значение по адресу this, т.е. объект А. Возвращать ссылку на объект необходимо для реализации выражения вида A*B*C.
Следует отметить, что если в описании класса dek_koord присутствуют объявления двух функций перегрузки операции *:
class dek_koord
{ . . .
dek_koord operator*(const dek_koord );
dek_koord &operator*(const dek_koord &);
. . .
};
то возникает ошибка. Аналогично ошибка будет, если одна из функций является компонентой класса, а другая − глобальной функцией.
Если возвращаемое значение функции operator является ссылкой, то в этом случае возвращаемое значение не может быть автоматической или статической локальной переменной.
Рассмотрим фрагмент программы, в которой функция operator* является глобальной.
class dek_koord
{ int x,y; // декартовы координаты точки
public:
. . .
int read_x(); // возвращает компоненту x
int read_y(); // возвращает компоненту y
void write_x(int); // модифицирует компоненту x
void write_y(int); // модифицирует компоненту y
. . .
};
int dek_koord::read_x(){return x;}
int dek_koord::read_y(){return y;}
void dek_koord::write_x(int a){x=a;}
void dek_koord::write_y(int a){y=a;}
dek_koord operator*(dek_koord a,dek_koord b) // перегрузка операции *
{ dek_koord tmp; // функция operator − глобальная
tmp.write_x(a.read_x()*b.read_x());
tmp.write_y(a.read_y()*b.read_y());
return tmp;
}
В глобальной функции operator* доступ к private данным локального объекта tmp возможен через public-функции этого объекта, либо данные класса должны иметь атрибут public, что не отвечает принципу инкапсуляции. Кроме того, если функция operator является friend-функцией некоторого класса, то она имеет доступ к private-компонентам этого класса. Это будет рассмотрено несколько ниже.
Далее приведен пример еще одной программы перегрузки оператора «- » для использования его при вычитании из одной строки другой.
#include <iostream>
using namespace std;
#include <string.h>
class String
{ char str[80]; // локальная компонента
public: // глобальные компоненты
void init (char *s); // функция инициализации
int operator - (String s_new); // прототип функции operator
} my_string1, my_string2; // описание двух объектов класса String
void String::init (char *s) // функция обеспечивает копирование
// строки аргумента(s) в строку-компоненту
{ strcpy(str,s);} // (str) класса String
int String::operator - (String s_new) // перегрузка оператора – (вычитания
// строк)
{ for (int i=0; str[i]==s_new.str[i]; i++)
if (!str[i]) return 0;
return str[i] - s_new.str[i];
}
int main()
{ char s1[51], s2[51];
cout <<"Введите первую строку (не более 80 символов) :" <<endl;
cin >>s1;
cout<<" Введите вторую строку (не более 80 символов) : "<<endl;
cin>>s2;
my_string1.init(s1); //инициализация объекта my_string1
my_string2.init(s2); //инициализация объекта my_string2
cout <<"\nString1 - String2 = "; // вывод на экран разности двух строк
cout << my_string1 - my_string2 << endl;
return 0;
}
Результат работы программы:
Введите первую строку (не более 80 символов) :
overload
Введите вторую строку (не более 80 символов) :
function
String1 – String2 = 9
При перегрузке бинарного оператора с использованием компоненты-функции ей передается в качестве параметра только один аргумент. Второй аргумент получается посредством использования неявного указателя this на объект, компоненты которого модифицируются.