gpt4 book ai didi

c++ - 向可变参数模板类添加回调 - 不可能?

转载 作者:太空宇宙 更新时间:2023-11-04 11:58:48 24 4
gpt4 key购买 nike

我正在尝试制作一个模板化类,它有一个 Add 方法,该方法将函数回调附加到该类,所以当我可以使用指定的参数列表从那里调用它时。它编译得很好,除了我调用回调的部分。它只是不接受参数,我已经尝试了我能想到的一切,但它仍然给我同样的“无法扩展参数包”错误。我正在使用 Visual Studio 2012 使用 Microsoft Visual C++ 编译器 2012 年 11 月 CTP (v120_CTP_Nov2012)这是示例来源: 模板 可变参数类 { 私有(private)的: void (*)($arguments...) callbackPtr;

public:
Variadic();
~Variadic();

void Attach(void (*callback)($arguments...));

void operator()($arguments... arguments);
};

然后我向它添加一个回调:

template<typename... $arguments>
void Variadic<$arguments...>::Attach(void (*callback)($arguments...))
{
callbackPtr = callback;
}

然后使用 () 运算符执行它:

template<typename... $arguments>
void Variadic<$arguments...>::operator()($arguments... arguments)
{
(callbackPtr)(arguments...);
}

在 main.cpp 中我做了一个小测试:

    void test(int testInt, float testFloat)
{
//DoNothing
}

int main()
{
Variadic<int, float> var; //create one that will have a callback that takes an int and a float argument
var.Attach(test); //attach test, which takes an int and a float as arguments
var(2, 3.0f); //try to call it
}

问题出现在我构建时 - 它在这一行给出了 2 个错误:(callbackPtr)(arguments...);错误是:

error C3546: '...' : there are no parameter packs available to expand error C2065: 'arguments' : undeclared identifier

起初我以为这是一个语法问题,我没有正确传递arguments...,但我尝试了所有可能的方法,它仍然给我同样的错误。我不能在谷歌中找到很多关于“参数包扩展”的信息。我可能做错了什么?我确定我在某种程度上错误地使用了 (callbackPtr)(arguments. ..); 调用,但不知道如何调用。

如有任何帮助,我们将不胜感激。

最佳答案

在我开始回答之前,你应该知道一些事情:

  • Microsoft VC++ 2012 年 11 月 CTP 不能很好地处理可变参数和函数指针/函数签名。在几乎所有情况下,都需要手动扩展它们。它很糟糕,但你必须忍受它,直到我们在 VS 和 VC++ 上投入的所有资金真正取得成果,并且我们得到一个具有其他编译器已经支持的大量 C++11 功能的编译器。

  • 传递函数指针并让编译器确定正确的类型比大多数人乍一看会猜测的要复杂一些。其中有很多类型的 诱惑 演绎和模板特化。

除此之外,基于函数指针和成员函数的回调需要很多模板魔法和很多有趣的功能,而不仅仅是使用 std::function<>。 .在向您展示我最终使用的解决方案之前,我强烈建议您使用 std::vector<std::function<[RETURNTYPE]( [PARAMS] )> > (或者只返回一次 std::function),以免自己因试图解决所有问题而头疼不已。在任何一种情况下,see my answer underneath @Insilico's for a Callback and Event system that works fine in GCC with variadic templates .

对于在 VC++ 中工作的版本,正如我之前所说,您必须手动破解各种定义,我最终创建了一个回调类和一个事件类来完成。它用于多个回调,但您可以简化 Event如果需要,类可以是单个附加/回调:

template<typename TFuncSignature>
class Callback;

/////////////////
/* 2 ARGUMENT */
/////////////////

template<typename R, typename T1, typename T2>
class Callback<R (T1, T2)> {
public:
typedef R (*TFunc)(void*, T1, T2);

const static size_t Arity = 2;

Callback() : obj(0), func(0) {}
Callback(void* o, TFunc f) : obj(o), func(f) {}

R operator()(T1 t1, T2 t2) const {
return (*func)(obj, t1, t2);
}

typedef void* Callback::*SafeBoolType;
operator SafeBoolType () const {
return func != 0? &Callback::obj : 0;
}

bool operator! () const {
return func == 0;
}

bool operator== ( const Callback<R (T1, T2)>& right ) const {
return obj == right.obj && func == right.func;
}

bool operator!= ( const Callback<R (T1, T2)>& right ) const {
return obj != right.obj || func != right.func;
}

private:
void* obj;
TFunc func;
};

namespace detail {
template<typename R, class T, typename T1, typename T2>
struct DeduceConstMemCallback2 {
template<R(T::*Func)(T1, T2) const> inline static Callback<R(T1, T2)> Bind(T* o) {
struct _ { static R wrapper(void* o, T1 t1, T2 t2) { return (static_cast<T*>(o)->*Func)(std::forward<T1>(t1, t2); } };
return Callback<R(T1, T2)>(o, (R(*)(void*, T1, T2)) _::wrapper);
}
};

template<typename R, class T, typename T1, typename T2>
struct DeduceMemCallback2 {
template<R(T::*Func)(T1, T2)> inline static Callback<R(T1, T2)> Bind(T* o) {
struct _ { static R wrapper(void* o, T1 t1, T2 t2) { return (static_cast<T*>(o)->*Func)(t1, t2)); } };
return Callback<R(T1, T2)>(o, (R(*)(void*, T1, T2)) _::wrapper);
}
};

template<typename R, typename T1, typename T2>
struct DeduceStaticCallback2 {
template<R(*Func)(T1, T2)> inline static Callback<R(T1, T2)> Bind() {
struct _ { static R wrapper(void*, T1 t1, T2 t2) { return (*Func)(t1), t2); } };
return Callback<R(T1, T2)>(0, (R(*)(void*, T1, T2)) _::wrapper);
}
};
}

template<typename R, class T, typename T1, typename T2>
detail::DeduceConstMemCallback2<R, T, T1, T2> DeduceCallback2(R(T::*)(T1, T2) const) {
return detail::DeduceConstMemCallback2<R, T, T1, T2>();
}

template<typename R, class T, typename T1, typename T2>
detail::DeduceMemCallback2<R, T, T1, T2> DeduceCallback2(R(T::*)(T1, T2)) {
return detail::DeduceMemCallback2<R, T, T1, T2>();
}

template<typename R, typename T1, typename T2>
detail::DeduceStaticCallback2<R, T1, T2> DeduceCallback2(R(*)(T1, T2)) {
return detail::DeduceStaticCallback2<R, T1, T2>();
}

template <typename T1, typename T2> class Event2 {
public:
typedef void(* TSignature)(T1, T2);
typedef Callback<void(T1, T2)> TCallback;
typedef std::vector<TCallback> InvocationTable;

protected:
InvocationTable invocations;

public:
const static int ExpectedFunctorCount = 2;

Event2 () : invocations() {
invocations.reserve( ExpectedFunctorCount );
}

Event2 ( int expectedfunctorcount ) : invocations() {
invocations.reserve( expectedfunctorcount );
}

template <void (* TFunc)(T1, T2)> void Add ( ) {
TCallback c = DeduceCallback2( TFunc ).template Bind< TFunc >( );
invocations.push_back( c );
}

template <typename T, void (T::* TFunc)(T1, T2)> void Add ( T& object ) {
Add<T, TFunc>( &object );
}

template <typename T, void (T::* TFunc)(T1, T2)> void Add ( T* object ) {
TCallback c = DeduceCallback2( TFunc ).template Bind< TFunc >( object );
invocations.push_back( c );
}

template <typename T, void (T::* TFunc)(T1, T2) const> void Add ( T& object ) {
Add<T, TFunc>( &object );
}

template <typename T, void (T::* TFunc)(T1, T2) const> void Add ( T* object ) {
TCallback c = DeduceCallback2( TFunc ).template Bind< TFunc >( object );
invocations.push_back( c );
}

void Invoke ( T1 t1, T2 t2 ) {
size_t i;
for ( i = 0; i < invocations.size(); ++i ) {
invocations[i]( t1, t2 );
}
}

void operator() ( T1 t1, T2 t2 ) {
size_t i;
for ( i = 0; i < invocations.size(); ++i ) {
invocations[i]( t1, t2 );
}
}

size_t InvocationCount ( ) {
return invocations.size( );
}

template <void (* TFunc)(T1, T2)> bool Remove ()
{ return Remove (DeduceCallback2(TFunc).template Bind<TFunc>()); }
template <typename T, void (T::* TFunc)(T1, T2)> bool Remove (T& object)
{ return Remove <T, TFunc>(&object); }
template <typename T, void (T::* TFunc)(T1, T2)> bool Remove (T* object)
{ return Remove (DeduceCallback2(TFunc).template Bind<TFunc>(object)); }
template <typename T, void (T::* TFunc)(T1, T2) const> bool Remove (T& object)
{ return Remove <T, TFunc>(&object); }
template <typename T, void (T::* TFunc)(T1, T2) const> bool Remove (T* object)
{ return Remove (DeduceCallback2(TFunc).template Bind<TFunc>(object)); }

protected:

bool Remove( TCallback const& target ) {
auto it = std::find(invocations.begin(), invocations.end(), target);
if ( it == invocations.end())
return false;
invocations.erase(it);
return true;
}

};

关于c++ - 向可变参数模板类添加回调 - 不可能?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15032594/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com