Je me réfère à l'article Implementing a Subject/Observer pattern with templatesmodèle à base de modèle Objet d'observateur - Dois-je utiliser static_cast ou dynamic_cast
Je l'ai fait quelques modifications et il est devenu le code comme suit.
template <class T, class A>
class Observer {
public:
Observer() {}
virtual ~Observer() {}
virtual void update(T& subject, A arg) = 0;
};
template <class T, class A>
class Subject
{
public:
Subject() {}
virtual ~Subject() {}
// Take note that, we didn't make the following functions as virtual,
// as we do not expect them to be overridden.
void attach(Observer<T, A> &observer) {
// Ensure no duplication.
std::vector<Observer<T, A> *>::const_iterator iterator = std::find(observers.begin(), observers.end(), &observer);
if (iterator == observers.end()) {
observers.push_back(&observer);
}
}
void dettach(Observer<T, A> &observer) {
std::vector<Observer<T, A> *>::const_iterator iterator = std::find(observers.begin(), observers.end(), &observer);
if (iterator != observers.end()) {
observers.erase(iterator);
}
}
void dettachAll() {
observers.clear();
}
void notify(A arg)
{
std::vector<Observer<T, A> *>::const_iterator it;
for (it = observers.begin(); it != observers.end(); it++) {
(*it)->update(*(static_cast<T *>(this)), arg);
}
}
private:
std::vector<Observer<T, A> *> observers;
};
Plus tard, je me rends compte que (*it)->update(*(static_cast<T *>(this)), arg);
a une limitation. Par exemple,
// cause compilation error in static_cast, as it cannot cast cat1 to animal.
class cat1 : public animal, public Subject<animal, int> {
public:
virtual void speak() {
notify(888);
}
};
class zoo1 : public Observer<animal, int> {
public:
zoo1() {
c.attach(*this);
c.speak();
}
virtual void update(animal& subject, int arg) {
cout << "zoo1 received notification " << arg << endl;
}
cat1 c;
};
je peux résoudre le problème en changeant le static_cast
à dynamic_cast
. Cependant, je ne suis pas sûr de tomber dans d'autres pièges? Ma conjecture sur l'intention originale de l'auteur en ayant static_cast
, est d'assurer la vérification de sécurité de type lors de la compilation.
Cela semble être répondu ici: [Dois-je utiliser la distribution dynamique dans le modèle d'observateur de sujet avec des modèles] (http://stackoverflow.com/questions/3975624/should-i-use-dynamic-cast-in-the -subject-observer-pattern-with-templates) – User