source: molecuilder/src/Patterns/Observer.hpp@ 973c03

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

Made the Observer RAII-Pattern more secure.

  • Property mode set to 100644
File size: 2.9 KB
Line 
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.
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.
22 *
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{
34 friend class Observable;
35public:
36 Observer();
37 virtual ~Observer();
38
39protected:
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
49 virtual void signOn(Observer *target, int priority=0);
50 virtual void signOff(Observer *target);
51
52protected:
53 virtual void update(Observable *publisher);
54 virtual void subjectKilled(Observable *publisher);
55
56 virtual void notifyAll();
57protected:
58// Observer mechanism is done from a static central place
59 /**
60 * Internal method.
61 * Do not call directly. Use OBSERVE macro instead
62 */
63 static void start_observer_internal(Observable *publisher);
64 /**
65 * Internal method.
66 * Do not call directly. Use OBSERVE macro instead
67 */
68 static void finish_observer_internal(Observable *publisher);
69
70private:
71 typedef std::multimap<int,Observer*> callees_t;
72 static std::map<Observable*, int> depth;
73 static std::map<Observable*,callees_t*> callTable;
74 static std::set<Observable*> busyObservables;
75
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};
86
87// extra macro is necessary to work with __LINE__
88#define PASTE(a,b) PASTE_HELPER(a,b)
89#define PASTE_HELPER(a,b) a ## b
90#define OBSERVE Observable::_Observable_protector PASTE(_scope_obs_protector_,__LINE__)(this)
91// deprecated macros from before RAII was used
92//#define START_OBSERVER Observable::start_observer_internal(this);do{do{}while(0)
93//#define FINISH_OBSERVER }while(0);Observable::finish_observer_internal(this)
94//#define RETURN_OBSERVER( retval ) do{Observable::finish_observer_internal(this); return (retval);}while(0)
95#endif /* OBSERVER_HPP_ */
Note: See TracBrowser for help on using the repository browser.