Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
Lections_rus.doc
Скачиваний:
31
Добавлен:
06.02.2016
Размер:
1.41 Mб
Скачать

4.1. Перегрузка операторов

На предыдущей лекции мы с вами встретились с функцией operator[],которая выполняла доступ к элементам вектора с проверкой диапазона.

Функции, которые имеют в своем названии слово «operator»имеют особенное название –функции перегрузки операторов.

Такие функции эффективно используются при создании пользователем собственных типов. Мы говорим о функции сложения нестандартных чисел, строк, матриц, дат, денег и тому подобное. Естественно и удобно использовать для выполнения этих действий операцию "+", а не определять соответствующие функции-компоненты.

Например, чтобы сложить два объекта fr1иfr2класса Fraction(Дробь), какой код Вы бы предпочли?

Fraction f3 = fr1 + fr2;

или

Fraction f3 = fr1.Add(fr2);

Или для класса string (строка), было бы удобно операцию конкатенации (сложения, соединения) строк s1 и s2 записать как

s1 + s2

но если Вы просто попытаетесь так сделать, то получите ошибку компиляции: "binary '+' : 'class string' does not define this operator"(класс string не определяет оператор +), другими словами компилятор не умеет применить эту операцию к типу, определенному пользователем.

Не огорчайтесь! :) Язык С++ (лучший язык программирования в мире!), дает возможность распространить действие практически любой стандартной операции на новые типы данных, вводимые пользователем. Сделать это позволяет механизм перегрузки стандартных операций.

Механизм перегрузки операций во многом схож с механизмом определения функций. Чтобы определить действие некоторой операции для нового пользовательского типа данных, нужно описать специальную функцию, называемую "операция-функция" (operator function). Формат определения операции-функции:

тип_возвращаемого_значения operator знак_операции

(список параметров операции-функции)

{

тело_операции-функции

}

Например, прототип операции-функции «+» для класса Fraction:

Fraction operator+(Fraction);

Теперь, чтобы сложить два объекта fr1 и fr2, достаточно просто записать выражение fr1 + fr2, которое интерпретируется как вызов функции fr1.operator+(fr2).

Определенная таким образом операция называется перегруженной (по-английски - overload), а сам механизм - перегрузкой или расширением действия стандартных операций языка С++.

4.1.1. Пример на перегрузку операторов

Давайте опишем класс CCurrency, который будем использовать для хранения и обработки денежных сумм.

class CCurrency

{

private:

long Dollars;

int Cents;

public:

CCurrency();

CCurrency (long Dol, int Cen);

void GetAmount (long *PDol, int *PCen);

void PrintAmount ();

void SetAmount (long Dol, int Cen);

CCurrency Sum(CCurrency aCurr);

}

# include <stdio.h>

# include " CCurrency.hpp"

// конструктор по умолчанию

CCurrency ::CCurrency() {

Dollars = 0;

Cents = 0;

}

// конструктор

CCurrency ::CCurrency (long Dol, int Cen) {

SetAmount (Dol, Cen);

}

// получить величину денежной суммы (в переменные PDol и PCen)

void CCurrency ::GetAmount (long *PDol, int *PCen) {

*PDol = Dollars;

*PCen = Cents;

}

// вывод денежной суммы на экран

void CCurrency ::PrintAmount () {

cout << '$' << Dollars << '.';

cout << Cents << '\n';

}

// назначить величину денежной суммы

void CCurrency ::SetAmount (long Dol, int Cen) {

Dollars = Dol + Cen/100;

Cents = Cen%100;

}

// сложить две денежные суммы

CCurrency CCurrency::Sum(CCurrency aCurr) {

CCurrency Temp;

Temp.SetAmount(Dollars + aCurr.Dollars, Cents + aCurr.Cents);

return Temp;

}

};

В классе CCurrency доллары и центы хранятся в виде отдельных велечин. При этом доллары имеют тип long.

Переменная Dollars объявлена как переменная типа long, с тем, чтобы она могла содержать максимально возможные целочисленные значения при использовании любого компилятора.

Класс содержит конструктор по умолчанию для установки в ноль начальных значений переменных (долларов и центов) и конструктор для присвоения этим переменным конкретных значений. Кроме того, имеется отдельная функция-член SetAmount, которая позволяет задать величину денежной суммы. Обратите внимание, что данные-члены класса (Dollars и Cents) объявлены со спецификатором доступа private, а значит, они недоступны в функции main и мы не можем их изменить напрямую, а только, используя функцию SetAmount. SetAmount при присвоении значений переменным Dollars и Cents преобразует их к нужному формату (величина центов - число в диапозоне [0,99]). Например, если вызвать эту функцию с параметрами (100, 180), то переменная Dollars получит значение 101, а Cents - 80 (т.е. 101 доллар и 80 центов). Также в классе описаны функции-члены GetAmount для получения значений переменных Dollars и Cents, PrintAmount для вывода денежной суммы на экран и Sum (в этом описании класса еще не использовали перегрузку операторов) для сложения двух денежных сумм.

Следующая программа использует класс CCurrencyдля подсчета суммы коммунальных услуг (вода, отопление, электричество) за месяц.

void main ()

{

CCurrency Heating(40,50); // плата за отопление

CCurrency Water(20,30); // плата за воду

CCurrency Electricity(25,67); // плата за электричество

CCurrency Total; // плата за коммунальные услуги

Total = Heating.Sum(Water);

Total = Total.Sum(Electricity);

Total.PrintAmount(); // выводим на экран общую сумму к оплате за услуги

}

Вместо задания функций-членов для арифметических операций с деньгами, можно перегрузить стандартные операторы С++. Это позволит выполнять арифметические действия, применяя выражения, традиционно используемые для встроенных типов данных.

Итак, для перегрузки оператора выполняется функция с именем operator, за которым следует обозначение операции. В нашем случае для сложения денежных сумм можно перегрузить оператор бинарного плюса (+),добавив в определении класса CCurrency (вместо функции CCurrency Sum(CCurrency aCurr); следующую функцию-член:

CCurrency operator+ (CCurrency aCurr) {

CCurrency Temp(Dollars + aCurr.Dollars, Cents + aCurr.Cents);

return Temp;

}

Функция-оператор определяется как public, чтобы ее могли использовать другие функции программы. Если такая функция задана, операцию сложения можно реализовать следующим образом:

CCurrency Amount1 (12,89);

CCurrency Amount2 (45, 91);

CCurrency Total;

Total = Amount1 + Amount2;

Компилятор языка С++ интерпретирует выражение Amount1 + Amount2 как

Amount1.operator+(Amount2);

Подобно стандартной операции сложения, выражение, содержащее более одного перегруженного оператора "+", вычисляется слева направо. Например, следующая программа использует перегруженный оператор для сложения величин, хранимых в трех объектах CCurrency.

void main ()

{

CCurrency Heating(40,50);

CCurrency Water(20,30);

CCurrency Electricity(25,67);

CCurrency Total;

Total = Heating + Water + Electricity;// правомерна строка

Total.PrintAmount();

}

Функцию operator+можно реализовать более эффективно, передавая ейссылкуна объект классаCCurrency, а не сам объект.

Передача ссылок исключает необходимость копирования объекта в локальный параметр, что особенно важно для объектов больших размеров.

Следующий фрагмент программы является окончательной версией функции operator+.

CCurrency operator+ (const CCurrency &aCurr) {

return CCurrency(Dollars + aCurr.Dollars, Cents + aCurr.Cents);

}

Использование спецификатора constпри объявлении параметра - гарантия того, что функция не изменит значение параметра.

Соседние файлы в предмете [НЕСОРТИРОВАННОЕ]