将抽象类的方法作为std::function传递
Passing the method of an abstract class as std::function
我有一些类似的代码(这只是一个片段,不是完全有效的代码(:
class AbstractClass {
public:
AbstractClass() {}
virtual ~AbstractClass() {}
virtual void doA() { std::cout << "doA1n"; };
virtual void doB() = 0;
};
class ImplClass : public AbstractClass {
public:
ImplClass() {}
virtual ~ImplClass() {}
virtual void doA() override { std::cout << "doA2n"; };
virtual void doB() override { std::cout << "doB2n"; };
};
现在,我想将对AbstractClass的成员方法的调用存储在一个std::函数中,如下所示:
int main() {
AbstractClass* aClass = new ImplClass();
std::function<void()> func = std::bind(&AbstractClass::doA, *aClass)
delete aClass;
return 0;
}
然而,根据编译器的说法,这是不起作用的,因为AbstractClass中存在纯虚拟方法。
编辑:完整错误供参考
In file included from source_file.cpp:2:0:
/usr/include/c++/5/functional: In instantiation of ‘struct std::_Bind_helper<false, void (AbstractClass::*)(), AbstractClass&>’:
/usr/include/c++/5/functional:1462:5: required by substitution of ‘template<class _Func, class ... _BoundArgs> typename std::_Bind_helper<std::__is_socketlike<_Func>::value, _Func, _BoundArgs ...>::type std::bind(_Func&&, _BoundArgs&& ...) [with _Func = void (AbstractClass::*)(); _BoundArgs = {AbstractClass&}]’
source_file.cpp:26:72: required from here
/usr/include/c++/5/functional:1445:71: error: invalid abstract parameter type ‘AbstractClass’
typedef _Bind<__func_type(typename decay<_BoundArgs>::type...)> type;
^
source_file.cpp:5:7: note: because the following virtual functions are pure within ‘AbstractClass’:
class AbstractClass {
^
source_file.cpp:10:18: note: virtual void AbstractClass::doB()
virtual void doB() = 0;
^
source_file.cpp: In function ‘int main()’:
source_file.cpp:26:72: error: no matching function for call to ‘bind(void (AbstractClass::*)(), AbstractClass&)’
std::function<void()> func = std::bind(&AbstractClass::doA, *aClass)
^
In file included from source_file.cpp:2:0:
/usr/include/c++/5/functional:1462:5: note: candidate: template<class _Func, class ... _BoundArgs> typename std::_Bind_helper<std::__is_socketlike<_Func>::value, _Func, _BoundArgs ...>::type std::bind(_Func&&, _BoundArgs&& ...)
bind(_Func&& __f, _BoundArgs&&... __args)
^
/usr/include/c++/5/functional:1462:5: note: substitution of deduced template arguments resulted in errors seen above
/usr/include/c++/5/functional:1490:5: note: candidate: template<class _Result, class _Func, class ... _BoundArgs> typename std::_Bindres_helper<_Result, _Func, _BoundArgs>::type std::bind(_Func&&, _BoundArgs&& ...)
bind(_Func&& __f, _BoundArgs&&... __args)
^
/usr/include/c++/5/functional:1490:5: note: template argument deduction/substitution failed:
source_file.cpp:26:72: note: couldn't deduce template parameter ‘_Result’
std::function<void()> func = std::bind(&AbstractClass::doA, *aClass)
^
我的问题有什么解决办法吗?到目前为止我还没有找到。
您的代码通过值绑定aClass
指向的对象。你想通过引用绑定它:
std::function<void()> func = std::bind(&AbstractClass::doA, std::ref(*aClass));
或者直接绑定aClass
,指针也可以用于隐式this
:
std::function<void()> func = std::bind(&AbstractClass::doA, aClass);
[实际示例]
目前最好的建议是支持lambdas而不是std::bind
,自从lambdas问世以来,这在标准中被视为某种时代错误。
Lambdas:
- 更明确
- 效率更高
- 不那么令人困惑
- 不需要像
std::ref
和std::cref
这样的复杂结构
示例:
#include <functional>
#include <iostream>
class AbstractClass {
public:
AbstractClass() {}
virtual ~AbstractClass() {}
virtual void doA() { std::cout << "doA1n"; };
virtual void doB() = 0;
};
class ImplClass : public AbstractClass {
public:
ImplClass() {}
virtual ~ImplClass() {}
virtual void doA() override { std::cout << "doA2n"; };
virtual void doB() override { std::cout << "doB2n"; };
};
int main() {
AbstractClass* aClass = new ImplClass();
// aClass is a pointer, and a copy of that pointer will be
// captured.
std::function<void()> func = [aClass]{ aClass->doA(); };
func();
delete aClass;
return 0;
}
相关文章:
- Confusion: decltype vs std::function
- 为什么 std::function 可以作为 std::not2 的参数?
- 'max'匹配'std::function<const int &(const int &, const int &)>'无过载
- 传递给std::function template的template参数究竟代表什么
- 绑定派生类方法C++从实例范围之外的分隔 std::function 变量调用
- 将函数包装器转换为 std::function
- 类型擦除的std::function与虚拟函数调用的开销
- C++ std::function 对于类 exept 的所有实例都是空的(只有 Visual2019 编译器问题)
- 如果模板没有可变参数,则 Lambda 被推导出为 std::function
- 将 lambda 表达式传递给 std::function in C++
- 模板类的部分模板专用化,如 std::function
- 可变参数模板参数扩展 类型为 std::function 的类成员
- 从 std::function in C++ 访问模板化 lambda
- 什么是 std::function::argument_type 的替代品?
- C++派生类重载函数(带有 std::function 参数)不可见
- 如何在 qi 符号表中使用 std::function
- 将 std::function 与模板一起使用
- C++ 事件管理器的回调,使用 std::function 和 std:bind 以及派生类作为参数
- 我需要在 std::function 上使用 unique_ptr 吗?
- 如何将函数指针从 std::function 传递到 Linux 克隆?