Скотт Майерс - Эффективное использование C++. 55 верных способов улучшить структуру и код ваших программ
- Название:Эффективное использование C++. 55 верных способов улучшить структуру и код ваших программ
- Автор:
- Жанр:
- Издательство:Литагент «ДМК»233a80b4-1212-102e-b479-a360f6b39df7
- Год:2006
- Город:Москва
- ISBN:5-94074-304-8
- Рейтинг:
- Избранное:Добавить в избранное
-
Отзывы:
-
Ваша оценка:
Скотт Майерс - Эффективное использование C++. 55 верных способов улучшить структуру и код ваших программ краткое содержание
Эта книга представляет собой перевод третьего издания американского бестселлера Effective C++ и является руководством по грамотному использованию языка C++. Она поможет сделать ваши программы более понятными, простыми в сопровождении и эффективными. Помимо материала, описывающего общую стратегию проектирования, книга включает в себя главы по программированию с применением шаблонов и по управлению ресурсами, а также множество советов, которые позволят усовершенствовать ваши программы и сделать работу более интересной и творческой. Книга также включает новый материал по принципам обработки исключений, паттернам проектирования и библиотечным средствам.
Издание ориентировано на программистов, знакомых с основами C++ и имеющих навыки его практического применения.
Эффективное использование C++. 55 верных способов улучшить структуру и код ваших программ - читать онлайн бесплатно ознакомительный отрывок
Интервал:
Закладка:
class Top {...};
class Middle: public Top {...};
class Bottom: public Middle {...};
Top *pt1 = new Middle; // преобразует Middle* в Top*
Top *pt2 = new Bottom; // преобразует Middle* в Bottom*
Const Top *pct2 = pt1; // преобразует Top* в const Top*
Эмулировать такие преобразования с помощью определяемых пользователем «интеллектуальных» указателей не просто. Для этого нужно, чтобы компилировался такой код:
Template
class SmartPtr {
public:
explicit SmartPtr(T *realPtr); // интеллектуальные указатели обычно
... // инициализируются встроенными
}; // указателями
SmartPtr pt1 = // преобразует SmartPtr
SmartPtr(new Middle); // в SmartPtr
SmartPtr pt2 = // преобразует SmartPtr
SmartPtr(new Bottom); // SmartPtr
SmartPtr pct2 = pt1;
Разные конкретизации одного шаблона не связаны каким-либо отношением, поэтому компилятор считает, что SmartPtr и SmartPtr – совершенно разные классы, не более связанные друг с другом, чем, например, vector и Widget. Чтобы можно было осуществлять преобразования между разными классами SmartPtr, необходимо явно написать соответствующий код. В приведенном выше примере каждое предложение создает новый объект интеллектуального указателя, поэтому для начала сосредоточимся на написании конструкторов, которые будут вести себя так, как нам нужно. Ключевое наблюдение состоит в том, что невозможно написать сразу все необходимые конструкторы. В приведенной иерархии мы можем сконструировать SmartPtr из SmartPtr или SmartPtr, но если в будущем иерархия будет расширена, то придется добавить возможность конструирования объектов SmartPtr из других типов интеллектуальных указателей. Например, если мы позже добавим такой класс:
class BelowBottom: public Bottom {...};
то нужно будет поддержать создание объектов SmartPtr из SmartPtr, и, очевидно, не хотелось бы ради этого модифицировать шаблон SmartPtr.
В принципе, нам может понадобиться неограниченное число конструкторов. Поскольку шаблон может быть конкретизирован для генерации неограниченного числа функций, похоже, что нам нужен не конструктор-функция для SmartPtr, а конструктор-шаблон. Это пример шаблона функции-члена (часто называемого шаблонного члена), то есть шаблона, генерирующего функции-члены класса:
template
class SmartPtr {
public:
template // шаблонный член
SmartPtr(const SmartPtr& other); // для «обобщенного
... // конструктора копирования»
};
Здесь говорится, что для каждой пары типов T и U класс SmartPtr может быть создан из SmartPtr, потому что SmartPtr имеет конструктор, принимающий параметр типа SmartPtr. Подобные конструкторы, создающие один объект из другого, тип которого является другой конкретизацией того же шаблона (например, SmartPtr из SmartPtr), иногда называют обобщенными конструкторами копирования.
Обобщенный конструктор копирования в приведенном выше примере не объявлен с модификатором explicit. И это сделано намеренно. Преобразования типов между встроенными типами указателей (например, из указателя на производный класс к указателю на базовый класс) происходят неявно и не требуют приведения, поэтому разумно и для интеллектуальных указателей эмулировать такое поведение. Именно поэтому и не указано слово explicit в объявлении обобщенного конструктора шаблона.
Будучи объявлен описанным выше образом, обобщенный конструктор копирования для SmartPtr предоставляет больше, чем нам нужно. Да, мы хотим иметь возможность создавать SmartPtr из SmartPtr, но вовсе не просили создавать SmartPtr из SmartPtr, потому что это противоречит смыслу открытого наследования (см. правило 32). Мы также не хотим создавать SmartPtr из SmartPtr, потому что не существует неявного преобразования int* в double*. Каким-то образом мы должны сузить многообразие функций-членов, которые способен генерировать этот шаблон.
Предполагая, что SmartPtr написан по образцу auto_ptr и tr1::shared_ptr, то есть предоставляет функцию-член get, которая возвращает копию встроенного указателя, хранящегося в объекте «интеллектуального» указателя (см. правило 15), мы можем воспользоваться реализацией шаблонного конструктора, чтобы ограничить набор преобразований:
template
class SmartPtr {
public:
template
SmartPtr(const SmartPtr& other) // инициировать этот хранимый
:heldPtr(other.get()) {...} // указатель указателем, хранящимся
// в другом объекте
T *get() const { return heldPtr;}
...
private: // встроенный указатель,
T *heldPtr; // хранящийся в «интеллектуальном»
};
Мы используем список инициализации членов, чтобы инициализировать член данных SmartPtr типа T* указателем типа U*, который хранится в Smart-Ptr. Этот код откомпилируется только тогда, когда существует неявное преобразование указателя U* в T*, а это как раз то, что нам нужно. Итак, SmartPtr теперь имеет обобщенный копирующий конструктор, который компилируется только тогда, когда ему передается параметр совместимого типа.
Использование шаблонных функций-членов не ограничивается конструкторами. Еще одно полезное применение таких функций – поддержка присваивания. Например, класс shared_ptr из TR1 (см. правило 13) поддерживает конструирование из всех совместимых встроенных указателей, tr1::shared_ptr, auto_ptr и tr1::weak_ptr (см. правило 54), а также наличие в правой части оператора присваивания объекта любого из этих типов, кроме tr1::weak_ptr. Ниже приведен фрагмент спецификации TR1 для tr1::shared_ptr; обратите внимание, что при объявлении параметров шаблона используется ключевое слов class, а не typename. Как объясняется в правиле 42, в данном контексте они означают одно и то же.
template class shared_ptr {
public:
template // конструирует из
explicit shared_ptr(Y *p); // любого совместимого
template // встроенного указателя,
shared_ptr(shared_ptr const& r); // shared_ptr,
template // weak_ptr или
explicit shared_ptr(weak_ptr const& r); // auto_ptr
template
explicit shared_ptr(auto_ptr& r);
template // присваивает
explicit shared_ptr& operator=(shared_ptr const& r); // любой
template // совместимый
explicit shared_ptr& operator=(auto_ptr const& r); // shared_ptr или
... // auto_ptr
};
Все эти конструкторы объявлены как explicit, за исключением обобщенного конструктора копирования. Это значит, что неявные преобразования от одного типа shared_ptr к другому допускаются, но неявные преобразования от встроенного указателя или другого «интеллектуального» указателя не допускаются. (Явные преобразования – например, с помощью приведения типов – разрешены). Также интересно отметить, что при передаче объекта auto_ptr конструктору tr1::shared_ptr и оператору присваивания параметр указывается без модификатора const, тогда как передаваемые параметры типа tr1::shared_ptr и tr1::weak_ptr константны. Это следствие того факта, что в отличие от других классов объекты auto_ptr модифицируются при копировании (см. правило 13).
Читать дальшеИнтервал:
Закладка: