-1

我有 2 个不同的应用程序,一个发送者和一个接收者。发送者将向接收者发送消息,接收者将解码消息并打印到控制台。但是,我不断收到分段错误错误。

发送方和接收方应用程序都具有相同的 TestContainer.h 和 TestContainer.cpp。

铸造方法

template<class To,class From>To cast(From v)
{
        return static_cast<To>(static_cast<const void*>)(v);
}

发件人申请

int main()
{
    TestContainer tc;
    tc.setDesc("this is a message");
    const char* castedData = cast<const char*>(&tc);

    const TestContainer test_tc = cast<const TestContainer*>(castedData);

    // i get back "this is a message",so the casting is working
    cout << "message content: " << test_tc->getDesc() <<endl;

    TaoSender;      
    TaoSender.send(castedData);     

    return 1;
}

接收机申请

void push(const RtecEventComm::EventSet& events)
{
    const char* receivedData;
    events[0].data.any_value >>= receivedData;

    cout << "data received: " << receivedData << endl;

    const TestContainer rcv_tc = cast<const TestContainer*>(receivedData);

    cout << "message content: " << rcv_tc->getDesc() <<endl; // error(segmentation fault)
}

TestContainer.h 和 TestContainer.cpp

class TestContainer{
    public 
        TestContainer();
        virtual ~TestContainer();
        const std:string& getDesc () const {
            return desc;
        }
        void setDesc(const std::string& desc) {
            this->desc = desc;
        }
    private
        std::string desc;
}



#include TestContainer.h

TestContainer::TestContainer(){}
TestContainer::~TestContainer(){}

发送方的castedData值和接收方的receivedData的值是一样的,所以我猜消息发送是正确的。

但是,在Receiver处,将 receivedData 缓冲区转换为 Testcontainer 指针并尝试访问 desc 后,我收到分段错误错误。

我还尝试在Sender中转换回 Testcontainer ,并且可以访问 desc。那么我错过了什么?

4

1 回答 1

1

尽管您问题中的代码包含错误,但如果我对您的理解正确,您正在尝试将对象的表示通过管道或其他通道推送到另一个进程,在那里您重建对象并尝试使用它。

您可以使用“Plain Old Data”对象执行此操作,但TestContainer该类具有虚拟方法,因此它不是 POD。在 C++ 的大多数实现中,类型的对象TestContainer将包含一个 vtable 指针,该指针指向在进程地址空间内分配的表。这个表的地址在不同的进程中可能会有所不同。TestContainer因此,当您尝试调用虚拟方法时,将对象的表示逐字节复制到另一个进程将导致崩溃。

TestContainer实例还包含一个std::string. Astring分配存储来保存字符串内容,并且该存储将在对象本身之外。传输string作为的一部分的字节表示会TestContainer导致在 a 的另一端进行重建,string它保存指向另一个进程中的内存地址的指针。在接收器中,该内存地址将引用其他内容。因此,string遗嘱不会被成功重建。

要在进程之间发送对象,您需要以独立于其中包含的内存地址的形式正确传输它们(无论是 vtable 指针、指针成员还是不透明成员的指针成员,例如 a string)。将数据转换为这种形式的过程称为序列化。有一些库可以帮助解决这个问题,但标准库中没有。

于 2016-11-18T10:05:44.367 回答