Название и классификация паттерна

Одиночка - паттерн, порождающий объекты.

Паттерн Singleton

Назначение

Гарантирует, что у класса есть только один экземпляр, и предоставляет к нему

глобальную точку доступа.

Мотивация

Для некоторых классов важно, чтобы существовал только один экземпляр. Хотя

в системе может быть много принтеров, но возможен лишь один спулер. Должны

быть только одна файловая система и единственный оконный менеджер. В циф-

ровом фильтре может находиться только один аналого-цифровой преобразователь

(АЦП). Бухгалтерская система обслуживает только одну компанию.

Как гарантировать,что у класса есть единственный экземпляр и что этот эк-

земпляр легко доступен? Глобальная переменная дает доступ к объекту, но не за-

прещает инстанцировать класс в нескольких экземплярах.

Более удачное решение - сам класс контролирует то, что у него есть только

один экземпляр, может запретить создание дополнительных экземпляров, пере-

хватывая запросы на создание новых объектов, и он же способен предоставить

доступ к своему экземпляру. Это и есть назначение паттерна одиночка.

Применимость

Используйте паттерн одиночка, когда:

а должен быть ровно один экземпляр некоторого класса, легко доступный

всем клиентам;

а единственный экземпляр должен расширяться путем порождения подклас-

сов, и клиентам нужно иметь возможность работать с расширенным экземп-

ляром без модификации своего кода.

Структура

Участники

a Singleton- одиночка:

- определяет операцию Instance, которая позволяет клиентам получать

доступ к единственному экземпляру. Instance - это операция класса, то

есть метод класса в терминологии Smalltalk и статическая функция-член

в C++;

- может нести ответственность за создание собственного уникального эк-

земпляра.

Порождающие паттерны

Отношения

Клиенты получают доступ к экземпляру класса Singleton только через его

операцию Instance.

Результаты

У паттерна одиночка есть определенные достоинства:

Q контролируемый доступ к единственному экземпляру. Поскольку класс

Singleton инкапсулирует свой единственный экземпляр, он полностью

контролирует то, как и когда клиенты получают доступ к нему;

а уменьшение числа имен. Паттерн одиночка - шаг вперед по сравнению с гло-

бальными переменными. Он позволяет избежать засорения пространства

имен глобальными переменными, в которых хранятся уникальные экземп-

ляры;

а допускает уточнение операций и представления. От класса Singleton мож-

но порождать подклассы, а приложение легко сконфигурировать экземп-

ляром расширенного класса. Можно конкретизировать приложение экземп-

ляром того класса, который необходим во время выполнения;

а допускает переменное число экземпляров. Паттерн позволяет вам легко изме-

нить свое решение и разрешить появление более одного экземпляра класса

Singleton. Вы можете применять один и тот же подход для управления чис-

лом экземпляров, используемых в приложении. Изменить нужно будет лишь

операцию, дающую доступ к экземплярукласса Singleton;

а большая гибкость, чем у операций класса. Еще один способ реализовать функ-

циональность одиночки - использовать операции класса, то есть статичес-

кие функции-члены в C++ и методы класса в Smalltalk. Но оба этих приема

препятствуют изменению дизайна, если потребуется разрешить наличие

нескольких экземпляров класса. Кроме того, статические функции-члены

в C++ не могут быть виртуальными, так что их нельзя полиморфно замес-

тить в подклассах.

Реализация

При использовании паттерна одиночка надо рассмотреть следующие вопросы:

а гарантирование единственного экземпляра. Паттерн одиночка устроен так,

что тот единственный экземпляр, который имеется у класса, - самый обыч-

ный, но больше одного экземпляра создать не удастся. Чаще всего для этого

прячут операцию, создающую экземпляры, за операцией класса (то есть за

статической функцией-членом или методом класса), которая гарантирует

создание не более одного экземпляра. Данная операция имеет доступ к пе-

ременной, где хранится уникальный экземпляр, и гарантирует инициализа-

цию переменной этим экземпляром перед возвратом ее клиенту. При таком

подходе можно не сомневаться,что одиночка будет создан и инициализи-

рован перед первым использованием.

В C++ операция класса определяется с помощью статической функции-чле-

на Instance класса Singleton. В этом классе есть также статическая

Паттерн Singleton Jill

переменная-член „instance, которая содержит указатель на уникальный

экземпляр.

Класс Singleton объявлен следующим образом:

class Singleton {

public:

static Singleton* Instance();

protected:

Singleton();

private:

static Singleton* „instance;

};

А реализация такова:

Singleton* Singleton::_instance = 0;

Singleton* Singleton::Instance () {

if (_instance == 0) {

_instance = new Singleton;

}

return „instance;

}

Клиенты осуществляют доступ к одиночке исключительно через функцию-

член Instance. Переменная „instance инициализируется нулем, а ста-

тическая функция-член Instance возвращает ее значение, инициализируя

ее уникальным экземпляром, если в текущий момент оно равно 0. Функция

Instance использует отложенную инициализацию: возвращаемое ей зна-

чение не создается и не хранится вплоть до момента первого обращения.

Обратите внимание, что конструктор защищенный. Клиент, который попы-

тается инстанцировать класс Singleton непосредственно, получит ошиб-

ку на этапе компиляции. Это дает гарантию, что будет создан только один

экземпляр.

Далее, поскольку „instance - указатель на объект класса Singleton, то

функция-член Instance может присвоить этой переменной указатель на

любой подкласс данного класса. Применение возможности мы увидим

в разделе ≪Пример кода≫.

О реализации в C++ скажем особо. Недостаточно определить рассматрива-

емый патерн как глобальный или статический объект, а затем полагаться на

автоматическую инициализацию. Тому есть три причины:

- мы не можем гарантировать, что будет объявлен только один экземпляр

статического объекта;

- у нас может не быть достаточно информации для инстанцирования лю-

бого одиночки во время статической инициализации. Одиночке могут

быть необходимы данные, вычисляемые позже, во время выполнения

программы;

- в C++ не определяется порядок вызова конструкторов для глобальных

объектов через границы единиц трансляции [ES90]. Это означает, что

Порождающие паттерны

между одиночками не может существовать никаких зависимостей. Если

они есть, то ошибок не избежать.

Еще один (хотя и не слишком серьезный) недостаток глобальных/статических

объектов в том, что приходится создавать всех одиночек, даже, если они не

используются. Применение статической функции-члена решает эту проблему.

В Smalltalk функция, возвращающая уникальный экземпляр, реализуется

как метод класса Singleton. Чтобы гарантировать единственность экземп-

ляра, следует заместить операциюnew. Получающийся класс мог бы иметь

два метода класса (в них Solelnstance - это переменная класса, которая

больше нигде не используется):

new

self error: 'не удается создать новый объект1

default

Solelnstance isNil ifTrue: [Solelnstance := super new].

^ Solelnstance

а порождение подклассов Singleton. Основной вопрос не столько в том, как опре-

делить подкласс, а в том, как сделать, чтобы клиенты могли использовать

его единственный экземпляр. По существу, переменная, ссылающаяся на

экземпляр одиночки, должна инициализироваться вместе с экземпляром

подкласса. Простейший способ добиться этого - определить одиночку, ко-

торого нужно применять в операции Instance класса Singleton. В раз-

деле ≪Пример кода≫ показывается, как можно реализовать эту технику с по-

мощью переменных среды.

Другой способ выбора подкласса Singleton - вынести реализацию опера-

ции Instance из родительского класса (например, MazeFactory) и помес-

тить ее в подкласс. Это позволит программисту на C++ задать класс оди-

ночки на этапе компоновки (скомпоновав программу с объектным файлом,

содержащим другую реализацию), но от клиента одиночка будет по-прежне-

му скрыт.

Такой подход фиксирует выбор класса одиночки на этапе компоновки, за-

трудняя тем самым его подмену во время выполнения. Применение услов-

ных операторов для выбора подкласса увеличивает гибкость решения, но

все равно множество возможных классов Singleton остается жестко ≪за-

шитым≫ в код. В общем случае ни тот, ни другой подход не обеспечивают

достаточной гибкости.

Ее можно добиться за счет использования реестра одиночек. Вместо того

чтобы задавать множество возможных классов Singleton в операции

Instance, одиночки могут регистрировать себя по имени в некотором всем

известном реестре.

Реестр сопоставляет одиночкам строковые имена. Когда операции Instance

нужен некоторый одиночка, она запрашивает его у реестра по имени. Начи-

нается поиск указанного одиночки, и, если он существует, реестр возвраща-

ет его. Такой подход освобождает Instance от необходимости ≪знать≫ все

Паттерн Singleton

возможные классы или экземпляры Singleton. Нужен лишь единый для

всех классов Singleton интерфейс, включающий операции с реестром:

class Singleton {

public:

static void Register(const char* name, Singleton*);

static Singleton* Instance ().;

protected:

static Singleton* Lookup(const char* name);

private:

static Singleton* „instance;

static List<NameSingletonPair>* „registry;

Операция Register регистрирует экземпляр класса Singleton под ука-

занным именем. Чтобы не усложнять реестр, мы будем хранить в нем спи-

сок объектов NameSingletonPair. Каждый такой объект отображает имя

на одиночку. Операция Lookup ищет одиночку по имени. Предположим, что

имя нужного одиночки передается в переменной среды:

Singleton* Singleton::Instance () {

if („instance == 0) {

const char* singletonName = getenv("SINGLETON");

// пользователь или среда предоставляют это имя на стадии

// запуска программы

_instance = Lookup(singletonName);

// Lookup возвращает 0, если такой одиночка не найден

return „instance;

В какой момент классы Singleton регистрируют себя? Одна из возмож-

ностей - конструктор. Например, подкласс MySingleton мог бы работать

так:

MySingleton::MySingleton() {

Singleton::Register("MySingleton", this);

}

Разумеется, конструктор не будет вызван, пока кто-то не инстанцирует

класс, но ведь это та самая проблема, которую паттерн одиночка и пытается

разрешить! В C++ ее можно попытаться обойти, определив статический эк-

земпляр класса My Single ton. Например, можно вставить строку

static MySingleton theSingleton;

в файл, где находится реализация MySingleton.

Теперь класс Singleton не отвечает за создание одиночки. Его основной

обязанностью становится обеспечение доступа к объекту-одиночке из

Порождающие паттерны

любой части системы. Подход, сводящийся к применению статического

объекта, по-прежнему имеет потенциальный недостаток: необходимо созда-

вать экземпляры всех возможных подклассов Singleton, иначе они не бу-

дут зарегистрированы.

Пример кода

Предположим, нам надо определить класс MazeFactory для создания лаби-

ринтов, описанный на стр. 99. MazeFactory определяет интерфейс для построения

различных частей лабиринта. В подклассах эти операции могут переопределять-

ся, чтобы возвращать экземпляры специализированных классов продуктов, на-

пример объекты BombedWall, а не просто Wall.

Существенно здесь то, что приложению Maze нужен лишь один экземпляр

фабрики лабиринтов и он должен быть доступен в коде, строящем любую часть

лабиринта. Тут-то паттерн одиночка и приходит на помощь. Сделав фабрику

MazeFactory одиночкой, мы сможем обеспечить глобальную доступность объек-

та, представляющего лабиринт, не прибегая к глобальным переменным.

Для простоты предположим, что мы никогда не порождаем подклассов от

MazeFactory. (Чуть ниже будет рассмотрен альтернативный подход.) В C++ для

того, чтобы превратить фабрику в одиночку, мы добавляем в класс MazeFactory

статическую операцию Instance и статический член _instance, в котором бу-

дет храниться единственный экземпляр. Нужно также сделать конструктор защи-

щенным, чтобы предотвратить случайное инстанцирование, в результате которо-

го будет создан лишний экземпляр:

class MazeFactory {

public:

static MazeFactory* Instance();

// здесь находится существующий интерфейс

protected:

MazeFactory();

private:

static MazeFactory* „instance;

};

Реализация класса такова:

MazeFactory* MazeFactory::_instance = 0;

MazeFactory* MazeFactory::Instance 0 {

if (_instance == 0) {

_instance = new MazeFactory;

}

return _instance;

}

Теперь посмотрим, что случится, когда у MazeFac tory есть подклассы и опре-

деляется, какой из них использовать. Вид лабиринта мы будем выбирать с по-

мощью переменной среды, поэтому добавим код, которыйинстанцирует нужный

Паттерн Singleton

подкласс MazeFactory в зависимости от значения данной переменной. Лучше

всего поместить код в операцию Instance, поскольку она уже и так инстанциру-

ет MazeFactory:

MazeFactory* MazeFactory::Instance () {

if (_instance == 0) {

const char* mazeStyle = getenv("MAZESTYLE");

if (strcmp(mazeStyle, "bombed") == 0) {

„.instance = new BombedMazeFactory;

} else if (strcmp(mazeStyle, "enchanted") == 0) {

_instance = new EnchantedMazeFactory;

// ... другие возможные подклассы

} else { // по умолчанию

_instance = new MazeFactory;

}

}

return _instance;

}

Отметим, что операцию Instance нужно модифицировать при определении

каждого нового подкласса MazeFactory. В данном приложении это, может быть,

и не проблема, но для абстрактных фабрик, определенных в каркасе, такой под-

ход трудно назвать приемлемым.

Одно из решений - воспользоваться принципом реестра, описанным в разде-

ле ≪Реализация≫. Может помочь и динамическое связывание, тогда приложению

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

Известные применения

Примером паттерна одиночка в Smalltalk-80 [РагЭО] является множество из-

менений кода, представленное классом ChangeSet. Более тонкий пример - это

отношение между классами и их метаклассами. Метаклассом называется класс

класса, каждый метакласс существует в единственном экземпляре. У метакласса

нет имени (разве что косвенное, определяемое экземпляром), но он контролирует

свой уникальный экземпляр, и создать второй обычно не разрешается.

В библиотеке Interviews для создания пользовательских интерфейсов

[LCI+92] - паттерн одиночка применяется для доступа к единственным экземпля-

рам классов Session (сессия) и WidgetKit (набор виджетов). Классом Session

определяется главный цикл распределения событий в приложении. Он хранит

пользовательские настройки стиля и управляет подключением к одному или не-

скольким физическим дисплеям. WidgetKit - это абстрактная фабрика для

определения внешнего облика интерфейсных виджетов. Операция Widget-

Kit: : instance () определяет конкретный инстанцируемый подкласс WidgetKit

на основе переменной среды, которую устанавливает Session. Аналогичная опе-

рация в классе Session ≪выясняет≫, поддерживаются ли монохромные или цвет-

ные дисплеи, и соответственно конфигурирует одиночку Session.

Порождающие паттерны

Родственные паттерны

С помощью паттерна одиночка могут быть реализованы многие паттерны. См.

описание абстрактной фабрики, строителя и прототипа.

Наши рекомендации