我在使用此类的C ++中使用sigaction时遇到问题:
class SerialHandler{
private:
...
/* Action Handling */
struct sigaction receive_handler;
/* Called using the receive_handler*/
static void receive_function(int status);
/* Get byte from the Serial Port */
unsigned char receive_byte();
/* Buffer for Singalinterrupt on Reciving */
std::vector<char> in_char_buffer;
...
};
我真的需要使用中断,并且确实需要在中断/ Sigaction上使用成员函数(receive_function),因为我需要访问成员变量(vector in_char_buffer)。问题:您不能使用sigaction使用成员函数,因为您必须将正常函数传递给sigaction.sa_handler。
正如您所看到的,使用static
函数也不是替代方法,因为您将无法访问成员变量。包装也无济于事。您可以访问新对象,但不能访问特定对象。
有没有可以解决这个问题的替代方案?
答案 0 :(得分:1)
这样的事情怎么样:
#include <signal.h>
#include <string.h>
#include <vector>
using namespace std;
class SerialHandler;
// A Singleton class
class SerialHandlerManager
{
private:
SerialHandlerManager() {};
public:
static SerialHandlerManager* get_instance();
void register_handler( int signum, int fd, SerialHandler * handler );
private:
static void dispatch_signal( int signum, siginfo_t * info, void * context );
private:
static SerialHandlerManager* the_manager;
vector<SerialHandler *> handler_objects;
};
class SerialHandler
{
private:
friend class SerialHandlerManager;
void install_signal_handler( int signum )
{
(SerialHandlerManager::get_instance())->register_handler( signum, fd, this );
}
void receive_function( int status );
int fd; // file descriptor returned from opening the serial port
};
SerialHandlerManager* SerialHandlerManager::the_manager;
SerialHandlerManager* SerialHandlerManager::get_instance()
{
if( !the_manager )
the_manager = new SerialHandlerManager;
return the_manager;
}
void SerialHandlerManager::register_handler( int signum, int fd, SerialHandler * handler )
{
struct sigaction act;
memset( &act, 0, sizeof(struct sigaction));
act.sa_flags = SA_SIGINFO;
act.sa_sigaction = SerialHandlerManager::dispatch_signal;
handler_objects[fd] = handler;
sigaction( signum, &act, NULL );
}
void SerialHandlerManager::dispatch_signal( int signum, siginfo_t * info, void * context )
{
(get_instance()->handler_objects[info->si_fd])->receive_function( signum );
}
当然,您可以使用地图将文件描述符链接到对象。这绝不是全面的(或防弹的),而是在阅读你的问题时我想到的一个快速的想法。
祝你好运阿尔文