Выбрать главу

namespace LIB = Disney_Feature_Animation;

int main()

{

LIB::Arrayint ia(1024);

}

Еще более удобным является способ использования простого, неквалифицированного имени для обращения к объектам, определенным в некотором пространстве имен. Для этого существует директива using:

#include "IBM_Canada_Laboratory.h"

using namespace IBM_Canada_Laboratory;

int main()

{

// IBM_Canada_Laboratory::Matrix

Matrix mat(4,4);

// IBM_Canada_Laboratory::Array

Arrayint ia(1024);

// ...

}

Пространство имен IBM_Canada_Laboratory становится видимым в программе. Можно сделать видимым не все пространство, а отдельные имена внутри него (селективная директива using):

#include "IBM_Canada_Laboratory.h"

using namespace IBM_Canada_Laboratory::Matrix;

// видимым становится только Matrix

int main()

{

// IBM_Canada_Laboratory::Matrix

Matrix mat(4,4);

// Ошибка: IBM_Canada_Laboratory::Array невидим

Arrayint ia(1024);

// ...

}

Как мы уже упоминали, все компоненты стандартной библиотеки С++ объявлены внутри пространства имен std. Поэтому простого включения заголовочного файла недостаточно, чтобы напрямую пользоваться стандартными функциями и классами:

#include string

// ошибка: string невидим

string current_chapter = "Обзор С++";

Необходимо использовать директиву using:

#include string

using namespace std;

// Ok: видим string

string current_chapter = "Обзор С++";

Заметим, однако, что таким образом мы возвращаемся к проблеме “засорения” глобального пространства имен, ради решения которой и был создан механизм именованных пространств. Поэтому лучше использовать либо квалифицированное имя:

#include string

// правильно: квалифицированное имя

std::string current_chapter = "Обзор С++";

либо селективную директиву using:

#include string

using namespace std::string;

// Ok: string видим

string current_chapter = "Обзор С++";

Мы рекомендуем пользоваться последним способом.

В большинстве примеров этой книги директивы пространств имен были опущены. Это сделано ради сокращения размера кода, а также потому, что большинство примеров были скомпилированы компилятором, не поддерживающим пространства имен – достаточно недавнего нововведения С++. (Детали применения using-объявлений при работе с стандартной библиотекой С++ обсуждаются в разделе 8.6.)

В нижеследующих главах мы создадим еще четыре класса: String, Stack, List и модификацию Stack. Все они будут заключены в одно пространство имен – Cplusplus_Primer_3E. (Более подробно работа с пространствами имен рассматривается в главе 8.)

Упражнение 2.21

Дано пространство имен

namespace Exercize {

template class elemType

class Array { ... };

template class EType

void print (Array EType );

class String { ... }

template class ListType

class List { ... };

}

и текст программы:

int main() {

const int size = 1024;

ArrayString as (size);

Listint il (size);

// ...

ArrayString *pas = new ArrayString(as);

Listint *pil = new Listint(il);

print (*pas);

}

Программа не компилируется, поскольку объявления используемых классов заключены в пространство имен Exercise. Модифицируйте код программы, используя

(a) квалифицированные имена

(b) селективную директиву using

(c) механизм псевдонимов

(d) директиву using

2.8. Стандартный массив - это вектор

Хотя встроенный массив формально и обеспечивает механизм контейнера, он, как мы видели выше, не поддерживает семантику абстракции контейнера. До принятия стандарта C++ для программирования на таком уровне мы должны были либо приобрести нужный класс, либо реализовать его самостоятельно. Теперь же класс массива является частью стандартной библиотеки C++. Только называется он не массив, а вектор.

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

vectorint ivec(10);

vectorstring svec(10);

Есть два существенных отличия нашей реализации шаблона класса Array от реализации шаблона класса vector. Первое отличие состоит в том, что вектор поддерживает как присваивание значений существующим элементам, так и вставку дополнительных элементов, то есть динамически растет во время выполнения, если программист решил воспользоваться этой его возможностью. Второе отличие более радикально и отражает существенное изменение парадигмы проектирования. Вместо того чтобы поддержать большой набор операций-членов, применимых к вектору, таких, как sort(), min(), max(), find()и так далее, класс vector предоставляет минимальный набор: операции сравнения на равенство и на меньше, size() и empty(). Более общие операции, перечисленные выше, определены как независимые обобщенные алгоритмы.