- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
尝试实现观察者模式,我希望主题和观察者代表所有可能的事件类型(避免重复代码)。我还想允许类通过多重继承从主题继承多次,以便被多个事件观察到。一种解决方案是制作 register/notify/etc... 方法模板,使其“喜欢”不同的名称但具有相同的代码,并消除调用的歧义,而无需制作包装函数。
我想要的东西:
object obj;
obj.registerEvent<Event1>(observer1);
obj.registerEvent<Event2>(observer2);
obj.registerEvent<Event3>(observer3);
...
实际代码:
#include <algorithm>
#include <type_traits>
#include <set>
#include <gtest/gtest.h>
template<typename Event>
class Observer
{
public:
protected:
Observer() = default;
public:
virtual void update() = 0;
};
template<typename Event>
class Subject
{
private:
std::set<Observer<Event>*> m_observers;
protected:
Subject() = default;
public:
#define same_template \
template<class T, class = typename std::enable_if<std::is_same<T, Event>::value>::type>
same_template
void registerObserver(Observer<Event>* observer, Event* = 0)
{
m_observers.insert(observer);
}
same_template
void removeObserver(Observer<Event>* observer, Event* = 0)
{
m_observers.erase(observer);
}
same_template
void notifyObservers()
{
std::for_each(m_observers.begin(), m_observers.end(), [] (Observer<Event> *observer) {
observer->update();
});
}
};
TEST(PatternObserverTest, CompilationTest)
{
struct Move {};
struct Jump {};
struct Player : Subject<Move>, Subject<Jump> {
void move()
{
notifyObservers<Move>(); // ERROR: member found in multiple bases classes of different types
}
void jump()
{
notifyObservers<Jump>(); // ERROR: member found in multiple bases classes of different types
}
};
struct Level : Observer<Move>, Observer<Jump> {
void update()
{
}
};
Player player;
Level level;
player.template registerObserver<Move>(&level); // ERROR: member found in multiple bases classes of different types
}
我什至尝试添加一个 foo 参数 (Event*),如果它会破坏 ODR。
问题是,要选择好的方法覆盖。看起来 SFINAE 不适用于多重继承。我怎样才能让它发挥作用?
我的猜测:我认为编译器会生成一组所有可能的方法,然后通过实例化消除错误的候选者,然后如果正好有 1 个剩余方法,则调用它,否则会抛出错误。看起来编译器是这样做的:它创建了一组可能的方法,并且因为有不止一个候选者,所以可以立即推断出调用是不明确的,因为在实例化方法之前作用域是不同的,只看函数名而不看函数名函数签名。
我知道可以使用通用父类(super class)和强制转换(方法 2)或为每个类型生成唯一 ID(方法 3)来解决问题,但我想坚持使用 C++,因为语法原因无法编译和崩溃错误。
最佳答案
函数必须在相同的范围内才能进行重载解析,并且可以通过 using
声明将其引入 Player
范围:
struct Player : Subject<Move>, Subject<Jump>
{
using Subject<Move>::notifyObservers;
using Subject<Jump>::notifyObservers;
using Subject<Move>::registerObserver;
using Subject<Jump>::registerObserver;
void move()
{
notifyObservers<Move>();
}
void jump()
{
notifyObservers<Jump>();
}
};
关于c++ - 如何让SFINAE工作选择通过多重继承继承的模板方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57516529/
我使用的是 PHP 5.3 稳定版,有时会遇到非常不一致的行为。据我所知,在继承中,父类(super class)中的所有属性和方法(私有(private)、公共(public)和 protected
所以我一直在努力寻找正确的方法来让应该非常简单的继承发挥作用(以我想要的方式 ;)),但我失败得很惨。考虑一下: class Parent { public String name = "Pare
给定这些类: class Father { public Father getMe() { return this; } } class Child extends Father {
为什么最后打印“I'm a Child Class”。 ? public class Parent { String parentString; public Parent()
我知道有很多类似的问题对此有很多很好的答案。我试着看看经典的继承方法,或者那些闭包方法等。不知何故,我认为它们对我来说或多或少是“hack”方法,因为它并不是 javascript 设计的真正目的。
我已经使用表单继承有一段时间了,但没有对以下方法进行太多研究。只需创建一个新类而不是表单并从现有表单继承并根据需要将所需控件转换为 protected 。 Visual Studio 2010 设计器
我原以为下面的代码片段会产生编译错误,因为派生类不会有我试图在 pub_fun() 中访问的 priv_var。但是它编译了,我得到了下面提到的输出。有人可以解释这背后的理论吗? class base
继承的替代方案有哪些? 最佳答案 Effective Java:优先考虑组合而不是继承。 (这实际上也来自《四人帮》)。 他提出的情况是,如果扩展类没有明确设计为继承,继承可能会导致许多不恰当的副作用
我有2个类别:动物( parent )和狗(动物的“ child ”),当我创建一个 Animal 对象并尝试提醒该动物的名称时,我得到了 undefined ,而不是她的真名。为什么?(抱歉重复发帖
我试图做继承,但没想到this.array会像静态成员一样。我怎样才能让它成为“ protected /公开的”: function A() { this.array = []; } func
在创建在父类中使用的 lambda 时,我试图访问子类方法和字段。代码更容易解释: class Parent { List> processors; private void do
如果我有一个对象,我想从“ super 对象”“继承”方法以确保一致性。它们将是混合变量。 修订 ParentObj = function() { var self = this; t
class Base { int x=1; void show() { System.out.println(x); } } class Chi
目前我正在尝试几种不同的 Javascript 继承方法。我有以下代码: (“借用”自 http://www.kevlindev.com/tutorials/javascript/inheritanc
我在 .popin-foto 元素中打开一个 popin。当我尝试在同一元素中打开子类 popin 时,它不起作用。 代码 这是 parent function Popin(container, ti
我有以下两个类: class MyClass { friend ostream& operatorvalue +=1; return *this; } 现在
有没有办法完全忽略导入到 html 文件中的 header 中的 CSS 文件? 我希望一个页面拥有自己独立的 CSS,而不是从任何其他 CSS 源继承。 最佳答案 您可以在本地样式表中使用 !imp
Douglas Crockford似乎喜欢下面的继承方式: if (typeof Object.create !== 'function') { Object.create = functio
假设我有以下代码: interface ISomeInterface { void DoSomething(); void A(); void B(); } public
class LinkedList{ public: int data; LinkedList *next; }; class NewLinkedList: public Lin
我是一名优秀的程序员,十分优秀!