- •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.5. Перегрузка оператора []
Как было отмечено выше, функция operator может быть с успехом использована для доопределения операторов С++ (в основном арифметические, логические и операторы отношения). В то же время в С++ существуют некоторые операторы, не входящие в число перечисленных, но которые полезно перегружать. К ним относится оператор []. Его необходимо перегружать с помощью компоненты-функции, использование friend-функции запрещено. Общая форма функции operator[]() имеет вид
тип_возвр_значения имя_класса::operator [](int i)
{ тело функции}
Параметр функции необязательно должен иметь тип int , но он использован, так как operator[] в основном применяется для индексации. Рассмотрим пример программы с использованием перегрузки операции []:
#include <iostream>
using namespace std;
class massiv
{ float f[3];
public:
massiv(float i,float j,float k){f[0]=i; f[1]=j; f[2]=k;}
float operator[](int i)
{ return f[i];} // перегрузка оператора []
};
int main()
{ massiv ff(1,2,3);
double f;
int i;
cout << "введите номер индекса ";
cin >> i;
cout <<"f["<< i <<" ]= " << ff[i] << endl;
return 0;
}
В примере перегруженная функция operator[]() возвращает величину элемента массива, индекс которого передан в функцию в качестве параметра. Данная программа при небольшой модификации может позволить использовать оператор[] как справа, так и слева от оператора присваивания. Для этого необходимо, чтобы функция operator[]() возвращала не элемент, а ссылку на него.
#include <iostream>
using namespace std;
class massiv
{ float f[3];
public:
massiv(float i,float j,float k){f[0]=i; f[1]=j; f[2]=k;}
float &operator[](int i) // перегрузка оператора []
{ if(i<0 || i>2) // проверка на выход за границы массива
{ cout << “Выход за пределы массива”<<endl;
exit(1);
}
return f[i];
}
};
int main()
{ massiv ff(1,2,3);
int i;
cout << "введите номер индекса ";
cin >> i;
cout <<"f["<< i <<" ]= " << ff[i] << endl;
ff[i]=5; // если функция operator не возвращает ссылку,то компиля-
// тор выдает ошибку =' : left operand must be l-value
cout <<"f["<< i <<" ]= " << ff[i] << endl;
return 0;
}
Рассмотрим случай, когда функция operator возвращает не ссылку, а указатель на модифицируемое значение. Внесем некоторые изменения в рассмотренную выше программу.
class massiv
{ float f[3];
public:
. . .
float *operator[](int i) // перегрузка оператора []
{ . . .
return &f[i];
}
};
int main()
{ massiv ff(1,2,3);
. . .
*ff[i]=5; // функция operator возвращает указатель
cout <<"f["<< i <<" ]= " << *ff[i] << endl;
return 0;
}
В приведенном примере создается ложное впечатление, что ff является вектором указателей. Поэтому использование ссылки представляется более предпочтительным.
Приведем еще один пример программы, использующей перегрузку operator[].
// перегрузка функции operator[] на примере вычисления n!
#include <iostream>
using namespace std;
#include "values.h" // для определения константы MAXLONG
class fact
{ long l;
public:
long operator[](int); // перегрузка оператора []
};
long fact::operator[](int n)
{ long l;
for (int i=0; i<=n; i++) //выбор буквы из уменьшаемой строки
if(l>MAXLONG/i)
cerr<<"ОШИБКА факториал числа "<<n<<" больше "<<MAXLONG;
else l*=i;
return l;
}
int main()
{ fact f;
int i,k;
cout << "введите число k для нахождения факториала"
cin >> k;
for (i=1; i<=k; i++)
cout << i <<"! = " << f[i] << endl;
return 0;
}