Добавил:
Upload Опубликованный материал нарушает ваши авторские права? Сообщите нам.
Вуз: Предмет: Файл:
тп модуль.docx
Скачиваний:
1
Добавлен:
10.09.2019
Размер:
1.06 Mб
Скачать

Паттерны поведения

Кстати, всегда желательно фиксировать, какие операции класса Subject

инициируют обновления;

а как избежать зависимости протокола обновления от наблюдателя: модели

вытягивания и проталкивания. В реализациях паттерна наблюдатель субъ-

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

мацию о характере изменения. Она передается в виде аргумента операции

Update, и объем ее меняется в широких диапазонах.

На одном полюсе находится так называемая модель проталкивания (push

model), когда субъект посылает наблюдателям детальную информацию об

изменении независимо от того, нужно ли им это. На другом - модель вытя-

гивания (pull model), когда субъект не посылает ничего, кроме минимально-

го уведомления, а наблюдатели запрашивают детали позднее.

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

их наблюдателях, а в модели проталкивания предполагается, что субъект вла-

деет определенной информацией о потребностях наблюдателей. В случае

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

может снизиться, так как классы Sub j ect предполагают о классах Observer,

которые не всегда могут быть верны. С другой стороны, модель вытягивания

может оказаться неэффективной, ибо наблюдателям без помощи субъекта

необходимо выяснять, что изменилось;

а явное специфицирование представляющих интерес модификаций Эффек-

тивность обновления можно повысить, расширив интерфейс регистрации

субъекта, то есть предоставив возможность при регистрации наблюдателя

указать, какие события его интересуют. Когда событие происходит, субъект

информирует лишь тех наблюдателей, которые проявили к нему интерес.

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

им субъектам следующим образом:

void Subject::Attach(Observer*, Aspects interest);

где interest определяет представляющее интерес событие. В момент по-

сылки уведомления субъект передает своим наблюдателям изменившийся

аспект в виде параметра операции Update. Например:

void Observer::Update(Subject*, Aspects interest);

а инкапсуляция сложной семантики обновления. Если отношения зависимос-

ти между субъектами и наблюдателями становятся особенно сложными, то

может потребоваться объект, инкапсулирующий эти отношения. Будем на-

зывать его ChangeManager (менеджер изменений). Он служит для мини-

мизации объема работы, необходимой для того чтобы наблюдатели смогли

отразить изменения субъекта. Например, если некоторая операция влечет

за собой изменения в нескольких независимых субъектах, то хотелось бы,

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

все субъекты, дабы не ставить в известность одного и того же наблюдателя

несколько раз.

Паттерн Observer

У класса ChangeManager есть три обязанности:

- строить отображение между субъектом и его наблюдателями и предо-

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

нии. Это освобождает субъектов от необходимости хранить ссылки на

своих наблюдателей и наоборот;

- определять конкретную стратегию обновления;

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

На следующей диаграмме представлена простая реализация паттерна на-

блюдатель с использованием менеджера изменений ChangeManager. Име-

ется два специализированных менеджера. SimplechangeManager всегда

обновляет всех наблюдателей каждого субъекта, a DAGChangeManager обра-

батывает направленные ациклические графы зависимостей между субъектами

и их наблюдателями. Когда наблюдатель должен присматривать за несколь-

кими субъектами, предпочтительнее использовать DAGChangeManager. В этом

случае изменение сразу двух или более субъектов может привести к избыточ-

ным обновлениям. Объект DAGChangeManager гарантирует, что наблюдатель

в любом случае получит только одно уведомление. Если обновление одного

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

но объекта SimplechangeManager.

ChangeManager - это пример паттерна посредник. В общем случае есть

только один объект ChangeManager, известный всем участникам. Поэтому

полезен будет также и паттерн одиночка;

а комбинирование классов Subject и Observer, В библиотеках классов, которые на-

писаны на языках, не поддерживающих множественного наследования (на-

пример, на Smalltalk), обычно не определяются отдельные классы Subject

и Observer. Их интерфейсы комбинируются в одном классе. Это позво-

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

Паттерны поведения

и наблюдателя, без множественного наследования. Так, в Smalltalk интерфей-

сы Sub j ect и Observer определены в корневом классе Obj ect и потому до-

ступны вообще всем классам.

Пример кода

Интерфейс наблюдателя определен в абстрактном классе Observer:

class Subject;

class Observer {

public:

virtual -Observer ();

virtual void Update (Subject* theChangedSubject) = 0;

protected:

Observer ( ) ,-

};

При такой реализации поддерживается несколько субъектов для одного на-

блюдателя. Передача субъекта параметром операции Update позволяет наблюда-

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

Таким же образом в абстрактном классе Subject определен интерфейс

субъекта:

class Subject {

public:

virtual -Subject()

virtual void Attach(Observer*);

virtual void Detach(Observer*);

virtual void Notify();

protected:

Subject();

private:

List<0bserver*> *_observers;

};

void Subject::Attach (Observer* o) {

_observers->Append(o);

}

void Subject::Detach (Observer* o) {

_observers->Remove(o);

}

void Subject::Notify () {

Listlterator<0bserver*> i(„observers);

for (i.First (); !i.IsDone() ; i.NextO) {

i.Currentltemf)->Update(this);

}

}