source: molecuilder/src/Patterns/Observer.hpp@ bc1fbf

Last change on this file since bc1fbf was bc1fbf, checked in by Tillmann Crueger <crueger@…>, 16 years ago

Changed Observer to use RAII-style for locking changes.

  • Property mode set to 100644
File size: 2.7 KB
RevLine 
[03d7ac]1/*
2 * Observer.hpp
3 *
4 * Created on: Jan 19, 2010
5 * Author: crueger
6 */
7
8#ifndef OBSERVER_HPP_
9#define OBSERVER_HPP_
10
11#include <map>
12#include <set>
13
14/**
15 * Basic structure for the observer pattern
16 *
17 * Observers register themselves with the observables to be notified when something changes.
[bc1fbf]18 * In the Observable code that changes attributes should be started with OBSERVE;. This macro
19 * locks the observer mechanism while changes are done. At the end of the scope in which the
20 * macro was placed the lock is released. When the last lock is released all changes are
21 * propagated to the observers.
[8fc9b6]22 *
[03d7ac]23 * Each observerable can have sub-observables. When one of these sub-observables changes and
24 * notifies its observers the observable that contains them will also notify its observers.
25 * This passing on of updates is blocked, when the main-observable is in the process of
26 * updating many of its internal sub-observables. This means the update is not passed, if
27 * it is produced while the main-observable itself is within any Observation block.
28 */
29
30class Observable;
31
32class Observer
33{
[6997fa]34 friend class Observable;
[03d7ac]35public:
36 Observer();
37 virtual ~Observer();
38
[6997fa]39protected:
[03d7ac]40 virtual void update(Observable *publisher)=0;
41 virtual void subjectKilled(Observable *publisher)=0;
42};
43
44class Observable : public Observer {
45public:
46 Observable();
47 virtual ~Observable();
48
[bb9503]49 virtual void signOn(Observer *target, int priority=0);
[03d7ac]50 virtual void signOff(Observer *target);
51
[6997fa]52protected:
[03d7ac]53 virtual void update(Observable *publisher);
54 virtual void subjectKilled(Observable *publisher);
55
56 virtual void notifyAll();
57protected:
[6997fa]58// Observer mechanism is done from a static central place
[03d7ac]59 /**
60 * Internal method.
[bc1fbf]61 * Do not call directly. Use OBSERVE macro instead
[03d7ac]62 */
63 static void start_observer_internal(Observable *publisher);
64 /**
65 * Internal method.
[bc1fbf]66 * Do not call directly. Use OBSERVE macro instead
[03d7ac]67 */
68 static void finish_observer_internal(Observable *publisher);
69
70private:
[bb9503]71 typedef std::multimap<int,Observer*> callees_t;
[03d7ac]72 static std::map<Observable*, int> depth;
[bb9503]73 static std::map<Observable*,callees_t*> callTable;
[03d7ac]74 static std::set<Observable*> busyObservables;
75
[bc1fbf]76 // Structure for RAII-Style notification
77protected:
78 class _Observable_protector {
79 public:
80 _Observable_protector(Observable *);
81 ~_Observable_protector();
82 private:
83 Observable *protege;
84 };
85};
[03d7ac]86
87
[bc1fbf]88#define OBSERVE Observable::_Observable_protector _scope_obs_protector(this)
[03d7ac]89#define START_OBSERVER Observable::start_observer_internal(this);do{do{}while(0)
90#define FINISH_OBSERVER }while(0);Observable::finish_observer_internal(this)
[8fc9b6]91#define RETURN_OBSERVER( retval ) do{Observable::finish_observer_internal(this); return (retval);}while(0)
[03d7ac]92#endif /* OBSERVER_HPP_ */
Note: See TracBrowser for help on using the repository browser.