【问题标题】:C++ Determine type of member function pointer template constantC++确定成员函数指针模板常量的类型
【发布时间】:2012-02-02 12:49:42
【问题描述】:

下面的代码在 Visual Studio 2010 中为我编译和工作。

现在我想改进一件小事,但我就是找不到办法。我尝试了一些模板模板技巧等,但无济于事。

我想要下面这行代码:

Sender.AttachListener<XSelectionChanged, TestListener, &TestListener::OnSelectionChanged>(Listener);

看起来像

Sender.AttachListener<&TestListener::OnSelectionChanged>(Listener);

也就是说,我向 AttachListener 模板提供了一个常量“void (TClass::*TMethod)(TEvent& _rEvent)”。从这个常量的类型我想得到 TClass 和 TEvent 类型。

这可能吗?如果是,怎么做?

-马蒂亚斯

代码:

#include <stdio.h>
#include <tchar.h>
#include <map>
#include <iostream>


struct XEvent
{};

struct XSelectionChanged : public XEvent
{};

struct XValueChanged : public XEvent
{};

template<typename TEvent, typename TClass>
struct TMethodType
{
    typedef void (TClass::*MethodType)(TEvent& _rEvent);
};

template<typename TEvent, typename TClass, void (TClass::*TMethod)(TEvent& _rEvent)>
struct TBoundMethod
{
    static void Dispatch(TEvent& _rEvent, TClass* _pInstance)
    {
        (_pInstance->*TMethod)(_rEvent);
    }
};


class CEventHost
{
public:
    template <typename TEvent, typename TClass, void (TClass::*TMethod)(TEvent& _rEvent)>
    void AttachListener(TClass& _rInstance)
    {
        TBoundMethod<TEvent, TClass, TMethod>::Dispatch( TEvent(), &_rInstance );
        //m_EventHost.Attach( &TBoundMethod<TClass, TEvent, TMethod>::Dispatch, &_rInstance );
    }

    template <typename TEvent>
    void SendEvent(TEvent& _rEvent)
    {
        //m_EventHost.Send( _rEvent );
    }

protected:
    //SHost m_EventHost;
};

class TestListener
{
public:
    void OnSelectionChanged(XSelectionChanged& _rEvent)
    {
        printf("Selection changed!\n");
    }
};


int _tmain(int argc, _TCHAR* argv[])
{
    CEventHost Sender, Sender2;
    TestListener Listener;

    Sender.AttachListener<XSelectionChanged, TestListener, &TestListener::OnSelectionChanged>(Listener);
    Sender.SendEvent(XSelectionChanged());
    Sender2.SendEvent(XSelectionChanged());
    Sender.SendEvent(XValueChanged());

    getchar();

    return 0;
}

【问题讨论】:

    标签: c++ function templates pointers member


    【解决方案1】:

    是的,使用元函数来提取 TClass 和 TEvent,如下所示:

    template<typename T>
    class GetSignature {};
    
    template<typename TClass, typename TEvent>
    class GetSignature <void (TClass::*)(TEvent&)>
    {
    public:
      typedef TClass Class;
      typedef TEvent Event;
    };
    
    template <typename T>
    void AttachListener(typename GetSignature<T>::Class& _rInstance)
    {
        typedef GetSignature<T> Signature;
        typedef void (TClass::*MethodType)(TEvent&);
        TBoundMethod<
            typename Signature::Event,
            typename Signature::Class,
            MethodType
        >::Dispatch( TEvent(), &_rInstance );
        //m_EventHost.Attach( &TBoundMethod<TClass, TEvent, TMethod>::Dispatch, &_rInstance );
    }
    

    警告:未经测试;-)

    【讨论】:

    • 谢谢。不幸的是,“class GetSignature ”这一行是无效的语法:(如果我删除“TMethod”位,它就会变成有效的语法。但是,编译器不能再推断 TMethod。跨度>
    • 是的,当然 - 这是一个值,而不是一个类型。过失,我会想办法解决的。
    • 这不起作用。提供给 TBoundMethod 的 MethodType 不再是一个常量,而是一个类型。
    【解决方案2】:

    你需要std::functionboost::function,真的。

    【讨论】:

    • 似乎 std::function 是一个运行时的东西。我想要编译时间。 -- Boost 在这个项目中是没有选择的
    猜你喜欢
    • 2011-12-09
    • 1970-01-01
    • 2015-02-25
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2010-09-13
    • 2019-01-04
    • 1970-01-01
    相关资源
    最近更新 更多