如何在静态函数中使用成员函数数组?

时间:2018-07-23 15:09:12

标签: c++ static boost-thread member-function-pointers

我有一堂课,名为Channel_thread。如您所料,它的一个函数是线程化的,在此函数中,我想根据情况调用另一个成员函数。这就是为什么我使用了一组成员函数。


这是代码的样子:

Channel_thread.cpp:

#include <boost/bind.hpp>
#include "../include/Channel_thread.h"


Channel_thread::Channel_thread(Event_queue<std::deque<char>> *_serverQueue, Event_queue<std::deque<char>> *_messageQueue)
{
    serverQueue = _serverQueue;
    messageQueue = _messageQueue;

}

void Channel_thread::startThread()
{
    isRunning = true;

    t = new boost::thread(boost::bind(&Channel_thread::start, this));
}

void Channel_thread::start(void *data)
{
    auto *_this = (Channel_thread *)data;
    Message *messageReceived = NULL;

    while (_this->isRunning)
    {
        std::pair<std::string, std::deque<char>> p(_this->messageQueue->wait_and_pop());
        for (int index = 0; index != 4; index++)
        {
            if (_this->handlersIndexTab[index] == p.first)
            {
                messageReceived = _this->handlersTab[index](p.second);
                break ;
            }
        }
    }
}

Message *Channel_thread::channelSetupHandler(std::deque<char> bytes)
{
//Do stuff
}

Message *Channel_thread::channelStatusHandler(std::deque<char> bytes)
{
//Do stuff
}

Message *Channel_thread::channelCloseHandler(std::deque<char> bytes)
{
//Do stuff
}

Message *Channel_thread::streamSetupHandler(std::deque<char> bytes)
{
//Do stuff
}

Channel_thread.h

#include <boost/thread.hpp>
#include "Event_queue.h"
#include "Channel_setup.h"
#include "Channel_status.h"
#include "Channel_close.h"
#include "Stream_setup.h"

class Channel_thread {

    typedef Message *(Channel_thread::*fn)(std::deque<char>);


public:
    Channel_thread(Event_queue<std::deque<char>> *, Event_queue<std::deque<char>> *);

    static void start(void *);

    void startThread();

    Message *channelSetupHandler(std::deque<char>);
    Message *channelStatusHandler(std::deque<char>);
    Message *channelCloseHandler(std::deque<char>);
    Message *streamSetupHandler(std::deque<char>);

private:
    Event_queue<std::deque<char>> *messageQueue;
    Event_queue<std::deque<char>> *serverQueue;

    bool isRunning;
    boost::thread *t;

    fn handlersTab[4] = {channelSetupHandler, channelStatusHandler, channelCloseHandler, streamSetupHandler};
    std::string handlersIndexTab[4] = {"channel_setup", "channel_status", "channel_close", "stream_setup"};
};

我收到此错误:

/cygdrive/c/Users/foo/CLionProjects/Server_cpp/src/Channel_thread.cpp: In static member function 'static void Channel_thread::start(void*)':
/cygdrive/c/Users/foo/CLionProjects/Server_cpp/src/Channel_thread.cpp:35:69: error: must use '.*' or '->*' to call pointer-to-member function in '_this->Channel_thread::handlersTab[index] (...)', e.g. '(... ->* _this->Channel_thread::handlersTab[index]) (...)'
                 messageReceived = _this->handlersTab[index](p.second);

如您所见,我在数组中调用成员函数的方式似乎是错误的,可能是由于static上下文,我并不十分了解,所以我的问题是:

如何从静态函数调用存储在成员函数数组中的成员函数?

谢谢。

2 个答案:

答案 0 :(得分:1)

使用_this->handlersTab[index](p.second);有几个问题。

  1. 使用成员函数指针的语法与使用成员函数的语法不同。成员函数指针在被调用之前需要被取消引用。给一个成员函数指针mfPtr,使用的语法是(objectPtr->*mfPtr)(...)。有关更多详细信息,请参见https://en.cppreference.com/w/cpp/language/pointer指向成员函数的指针部分。

  2. 第二个问题是handlersTab不是static成员变量。要获得指向成员函数的指针,必须使用_this->handlersTab

一个可以进行函数调用的衬里是:

messageReceived = (_this->*(_this->handlersTab[index]))(p.second);

我建议使用两行来简化它。

fn handler = _this->handlersTab[index];
messageReceived = (_this->*handler)(p.second);

答案 1 :(得分:1)

如果您使用的是c ++ 17,最简单的解决方案是在功能标头中使用std :: invoke。

std::invoke(&Channel_thread::handlersTab[index], this, p.second);

确保handlersTab是静态的。 handlersIndexTab也应该如此。

根据您的情况,另一种方法是使用枚举而不是字符串作为消息,然后打开该枚举并直接调用所需的方法。如果消息完全由您控制,包括传输通道,这是最合适的。

如果在消息中使用描述性字符串很重要,那么我所见的另一种方法是打开消息的哈希值。 哈希必须是constexpr函数,以便可以在编译时计算大小写值。用法:

switch (hash(p.first))
{
case "channel_setup"_hash:
    channelSetupHandler(p.second)
    break;
...
default:
    // error handling
}

使用这种方法的优点是

  • 直接调用所需方法
  • 无需同步两个列表,实际上不需要功能列表或查找列表
  • 默认情况下可用于处理不良消息
  • 更透明的代码
  • 易于扩展