Типы оградительных сооружений в морском порту: По расположению оградительных сооружений в плане различают волноломы, обе оконечности...
Историки об Елизавете Петровне: Елизавета попала между двумя встречными культурными течениями, воспитывалась среди новых европейских веяний и преданий...
Топ:
Оценка эффективности инструментов коммуникационной политики: Внешние коммуникации - обмен информацией между организацией и её внешней средой...
Генеалогическое древо Султанов Османской империи: Османские правители, вначале, будучи еще бейлербеями Анатолии, женились на дочерях византийских императоров...
Особенности труда и отдыха в условиях низких температур: К работам при низких температурах на открытом воздухе и в не отапливаемых помещениях допускаются лица не моложе 18 лет, прошедшие...
Интересное:
Инженерная защита территорий, зданий и сооружений от опасных геологических процессов: Изучение оползневых явлений, оценка устойчивости склонов и проектирование противооползневых сооружений — актуальнейшие задачи, стоящие перед отечественными...
Наиболее распространенные виды рака: Раковая опухоль — это самостоятельное новообразование, которое может возникнуть и от повышенного давления...
Лечение прогрессирующих форм рака: Одним из наиболее важных достижений экспериментальной химиотерапии опухолей, начатой в 60-х и реализованной в 70-х годах, является...
Дисциплины:
2018-01-13 | 566 |
5.00
из
|
Заказать работу |
|
|
Связывание – подстановка в коды программы вызовов конкретных функций – методов класса. Имеет смысл только для производных классов.
Обычно компилятор имеет необходимую информацию для того, чтобы определить, какая функция имеется в виду. Например, если в программе встречается вызов obj.f(), компилятор однозначно выбирает функцию f() в зависимости от типа адресата obj. Если в программе используются указатели на экземпляры класса: ptr->f(), выбор функции - метода класса определяется типом указателя.
Если выбор функции выполняется на этапе компиляции, мы имеем дело со статическим связыванием.
В этом случае для указателя на базовый класс будет вызвана функция – метод базового класса, даже если указателю на базовый класс присвоить значение адреса экземпляра производного класса.
Если выбор функции выполняется на этапе выполнения программы, мы имеем дело с динамическим связыванием.
В этом случае если при выполнении программы указателю на базовый класс присвоить адрес экземпляра базового класса, будет вызван метод базового класса; если же указателю на базовый класс присвоить адрес экземпляра производного класса, будет вызван метод производного класса.
Виртуальные функции
По умолчанию для производных классов устанавливается статическое связывание. Если для каких-либо методов класса нужно использовать динамическое связывание, такие методы должны быть объявлены виртуальными.
Виртуальные функции:
- имеют в прототипе в базовом классе ключевое слово virtual;
- обязательно функции-члены класса:
- Во всех производных классах должны иметь такой же прототип (указание слова virtual в производных классах не обязательно).
|
Если какие-либо методы в производных классах имеют то же имя, что и в базовом классе, но другой список параметров, мы имеем дело с перегруженными функциями.
Пример: классы Точка и Окружность.
class Point{
protected:
int x, y;
public:
...
virtual void print(); };
class Circle: public Point{
private:
int rad;
public:
...
void print(); // можно virtual void print(); };
void Point::print()
{ cout << "Point (" << x << ", " << y << ")"; }
void Circle::print()
{ cout << "Circle with center in "; Point::print();
cout << "and radius " << rad;
}
Использование:
Point p1(3,5), p2(1,1), *pPtr;
Cicle c1(1), c2(p2, 1);
pPtr = &p1; pPtr->print(); // получим: Point (3, 5)
pPtr = &c2; pPtr->print(); // получим:
Circle with center in Point (1, 1) and radius 1
Пример использования динамического связывания: список
Наиболее часто динамическое связывание используется с контейнерными классами, содержащими указатель на базовый класс; в такие контейнерные классы можно включать информацию, относящуюся и к базовому, и к любым производным классам.
Рассмотрим пример – список, содержащий и точки, и окружности.
struct Item{
Point *info;
Item *next;
// конструктор
Item():info(NULL), next(NULL){}
Item(Point *p):info(p), next(NULL){} };
class List{
private:
Item *head;
public:
List():head(NULL){}
void insert(Point *p){p->next = head; head = p;}
void print(); };
void List::print()
{ for(Item *cur = head; cur; cur = cur->next){
cur->info->print();
cout << endl; } }
Использование класса:
List mylist;
Point *p = new Point(1,2);
mylist.insert(p);
p = new Cicle(1,2,1);
mylist.insert(p);
mylist.print();
получим:
Circle with center in Point (1, 2) and radius 1
Point (1, 2)
Итераторы используются для обхода контейнеров. По сути, итератор — это класс, содержащий указатель на определенный элемент контейнера, а также функции, с помощью которых можно получить итераторы, указывающие на другие элементы контейнера.
Получить итератор можно различными методами классов контейнеров, например итераторы, указывающие на первый элемент контейнера можно получить, вызвав функцию.begin() в классе контейнера, а вызвав в том же классе контейнера функцию.end(), можно получить указатель на место, находящееся за позицией последнего элемента контейнера.
|
Итератор можно инкрементировать и декрементировать, атак же разыменовывать, как указатель, при этом получая содержимое контейнера, на который он указывает.
Одним из главных преимуществ итераторов является то, что он не позволяет так легко, как с помощью обычного индексирования выйти за пределы конейнера: если при их использовании происходит выход за пределы контейнера, происходит исключение, которое можно отловить, используя средства языка C++. При использовании же индексирования, если произошел выход за пределы контейнера, можно получить, порой, очень трудно обнаружаемые ошибки. Однако, если алгоритм не подразумевает последовательного обхода контейнера (например, обход списка), лучше использовать индексирование — с ним код выглядит более понятным и наглядным.
list<int> l;
list<int>::iterator i = l.begin();
while (i!= l.end())
{
//do something
++i;
}
пример использования для обхода списка
vector<int> v;
for (vector<int>::iterator i = v.begin(); i < v.end(); ++i)
{
//do something
}
пример использования для обхода вектора
Стр 164
Поток — это общее название потока данных. В C++ поток представляет собой объект некоторого класса. Именно поэтому вы могли встретить в листингах потоковые объекты cin и cout. Разные потоки предназначены для представления разных видов данных.
Одним из аргументов в пользу потоков является простота использования. Каждый объект сам знает, как он должен выглядеть на экране. Это избавляет программиста от одного из основных источников ошибок. Другой причиной является то, что можно перегружать стандартные операторы и функции вставки (<<) и извлечения (>>) для работы с создаваемыми классами. Это позволяет работать с собственными классами как со стандартными типами, что, опять же, делает программирование проще и избавляет от множества ошибок. Оказывается, потоковый ввод/вывод, нужен. Потому что это лучший способ записывать данные в файл, лучший способ организации данных в памяти для последующего использования при вводе/выводе текста в окошках и других элементах графического интерфейса пользователя (GUI).
|
ИЕРАРХИЯ.
Потоковые классы имеют довольно сложную иерархическую структу-ру. Операция извлечения >> является методом класса istream, опера-ция вставки << — методом класса ostream. Оба этих класса являются наследниками ios. Некоторые манипуляторы описаны в IOMANIP, а некоторые классы для работы с объектами «в памяти» определены в STRSTREAM. Класс ios является базовым для всей иерархии. Он со-держит множество констант и методов, общих для операций ввода/ вывода любых видов. Класс istream содержит функции: get(), getline(), read() и перегружаемую операцию извлечения (>>). Класс ostream содержит функции: put(), write() и перегружаемую операцию вставки (<<). Класс iostream — наследник одновременно классов istream и ostream (пример множественного наследования). Его производные классы могут использоваться при работе с объектами – дисковые файлы, которые могут быть открыты одновременно для записи и чте-ния. Классы: istream_withassign, ostream_withassign,iostream_withas-sign — являются наследниками istream, ostream и iostream соответ-ственно. Они добавляют к этим классам операторы присваивания.
Для поддержки ввода и вывода на основе потоков используются билиотеки fstream:
· Ifstream – класс, с помощью которого осуществляется чтение из файла
· Ofstream- класс, с помощью которого осуществляется запись в файл
· Fstream – класс дя чтения и записи в файл.
Необходимая директива: #include <fstream>.
Работа с файлами предполагает след операции:
1. Создание потокового объекта
2. Открытие потока и связывание его с файлом
3. Осуществление чтения и записи
4. Закрытие файла.
Запись данных в текст файл:
Шаблоны типа для функций
|
|
Стр 215
|
|
Эмиссия газов от очистных сооружений канализации: В последние годы внимание мирового сообщества сосредоточено на экологических проблемах...
Состав сооружений: решетки и песколовки: Решетки – это первое устройство в схеме очистных сооружений. Они представляют...
Историки об Елизавете Петровне: Елизавета попала между двумя встречными культурными течениями, воспитывалась среди новых европейских веяний и преданий...
Общие условия выбора системы дренажа: Система дренажа выбирается в зависимости от характера защищаемого...
© cyberpedia.su 2017-2024 - Не является автором материалов. Исключительное право сохранено за автором текста.
Если вы не хотите, чтобы данный материал был у нас на сайте, перейдите по ссылке: Нарушение авторских прав. Мы поможем в написании вашей работы!