|
Программирование >> Синтаксис инициирования исключений
Ведение статистики класса Также несложно организовать накопление статистики об использовании операторов Type* и -> в статических переменных параметризованного класса. template <c1ass Type> class SPCS { private: Type* pointer; static int conversions; static int members; public: SPCS() : pointer(NULL) {} SPCS(Type* p) : pointer(p) {} operator Type*() { conversions++; return pointer; } Type* operator->() { members++; return pointer; } int Conversions() { return conversions; } int Members() { return members; } Глобальные переменные должны быть где-то определены. Обычно это делается в файле Foo.cpp: В файле Foo.cpp int Ptr<Foo>::conversions = 0; int Ptr<Foo>::members = 0; Разумеется, вы можете воспользоваться директивами #ifdef, чтобы это относилось только к отладочной версии. Ведение статистики объекта Мы подошли к более сложной теме. Возможно, ее следует отложить до знакомства с ведущими указателями (master pointers), однако умные указатели также могут вести статистику по отдельным объектам, а не по классу в целом. Задача не сводится к тому, чтобы сделать только что показанные переменные нестатическими (то есть по одному экземпляру переменных на указатель), поскольку мы (пока) не можем обеспечить однозначное соответствие между указателями и объектами. Вместо этого статистику придется хранить в самих объектах. Ниже приведен полезный вспомогательный класс, который можно создать на основе множественного наследования как производный от класса указываемого объекта и от класса умного указателя, знающего о его свойствах. Объявляя указатель другом, вы предоставляете ему доступ к защищенным членам классов, производных от Counter. class Counter { protected: Counter() : conversions(O), members(O) {} Counter(const Counter&) : conversions(O), members(O) {} Counter& operator=(const Counter&) { return *this; } public: int conversions; int members; int Conversions() { return conversions; } int Members() { return members; } template <c1ass Type> class SPOP { private: Type* pointer; public: SPOS() : pointer(NULL) {} SPOP(Type* f) : pointer(f) {} operator Type*() { pointer->conversions++; return pointer; } Type* operator->() { pointer->members++; return pointer; } На эту тему существует ряд вариаций, с некоторыми из них мы познакомимся при изучении ведущих указателей. Кэширование Иногда нельзя даже настаивать, чтобы объект физически находился в памяти все время, пока к нему нужен доступ. В следующем примере предполагается, что функция ReadObject() умеет использовать данные о местонахождении объекта на диске, чтобы создать экземпляр и занести его адрес в указатель pointer . Если при вызове операторов объект отсутствует в памяти, он автоматически считывается с диска. typedef unsigned long DiskAddress; Заменить нужными данными template <c1ass Type> class CP { private: DiskAddress record number; Type* pointer; void ReadObject(); Считывает объект с диска public: CP(DiskAddress da) : pointer(NULL), record number(da) {} CP(Type* f) : pointer(f), record number(f->RecordNumber()) {} operator Type*() if (pointer == NULL) this->ReadObject(); return pointer; Type* operator->() if (pointer == NULL) this->ReadObject(); return pointer; Подробно говорить о кэшировании преждевременно, поскольку приходится учитывать множество проблем, к которым мы еще не готовы. Если вы хотите гарантировать, что читается лишь одна копия объекта независимо от того, сколько различных объектов на нее ссылается, или чтобы объект уничтожался сразу после завершения работы операторов, вам придется подождать следующих глав, посвященных ведущим указателям и управлению памятью. Тем не менее, в простых ситуациях с одним считыванием, в которых может существовать несколько копий объекта, такая методика достаточно хорошо работает и с простыми умными указателями. Кэширующие указатели используют один распространенный прием - они экономят несколько бит за счет объединения дискового адреса и адреса памяти в одной переменной класса. При этом используются два упрощающих предположения: 1 . Размер адреса памяти не более чем на один бит превышает размер дискового адреса. 2. Средства управления памятью, используемые оператором new, никогда не возвращают нечетный адрес. Если оба предположения верны, для хранения обоих адресов можно использовать одну переменную класса. Если младший бит установлен (то есть если адрес четный), остальные 31 бит определяют дисковый адрес. Когда младший бит сброшен, все 32 бита определяют адрес памяти. Если вам потребуется не только считывание, но и запись, объекту лучше знать свой собственный дисковый адрес, поскольку адрес, хранящийся в указателе, при считывании портится. За дымовой завесой кэширующих указателей прячется интересная концепция: умные указатели могут использоваться как общее средство для доступа к объектам независимо от того, где находится объект и существует ли он вообще. Углубляясь в джунгли C++, мы будем рассматривать эту концепцию под разными углами, пока она не превратится в один из принципов Дао, о которых я упоминал во вступительной главе.
|
© 2006 - 2025 pmbk.ru. Генерация страницы: 0
При копировании материалов приветствуются ссылки. |