4

我正在使用 libPoco 创建一个虚拟服务器来测试一些客户端代码。

class ServerRunnable: public Poco::Runnable {
  public:
ServerRunnable(StreamSocket conn) : conn(conn) {
}

void run(){
  string mess("Can you hear me?\n");
  try{
    this->conn.sendBytes(mess.c_str(), mess.size());
  } catch (Poco::Exception& ex){
    cerr << ex.displayText() << endl;
    return;
  }
  cerr << "The message has been sent." << endl;
}

void setConn(StreamSocket inConn){
  this->conn = inConn;
}
  private:
StreamSocket conn;
};


int main(int argc, char **argv){
  ServerSocket s;
  try{
    s.bind(8083, true);
  } catch (Exception &ex){
    cerr << ex.displayText() << endl;
    exit(1);
  }
  s.listen(124);

  Poco::ThreadPool Pool(10, 25, 60, 128);
  while(1){
    try{
      StreamSocket conn = s.acceptConnection();
      ServerRunnable serveIt(conn);

      Pool.start(serveIt);
    }  catch (Exception &ex){
      cerr << ex.displayText() << endl;
      Pool.joinAll();
      exit(1);
    }
  } 
  return 0;
 }

Poco::Runnable是一个抽象类,我很确定 run 是一个纯虚函数。Pool.start(serveIt)好像是打电话ServerRunnablerun。当我从控制台运行它时,我总是会pure virtual method called出错。但是,如果我在 gdb 中单步执行代码,那么我将成功接受来自客户端的连接并将数据发送给他们。ServerRunnable'srun不是纯虚函数,应该调用它。

libPoco 线程的代码示例位于http://pocoproject.org/slides/130-Threads.pdf

我也在想我可能在构造函数中调用纯虚方法,但构造函数中没有任何内容,我只是使用默认的析构函数。有没有办法确定调用纯虚函数的位置和对象?在gdb中?谢谢。

4

2 回答 2

7

问题很可能是您的对象在调用它的方法serverIt之前超出了范围。run

您无法控制线程何时运行,因此循环可能会在run调用您的类的方法之前进行迭代,但随后您的对象已被破坏,并且当然也破坏了虚函数表。

于 2013-06-28T05:35:09.993 回答
2
Pool.start()

启动一个线程并返回。这就是为什么在 PDF 中链接他们的线程池示例如下所示:

main()
    Poco::ThreadPool::defaultPool().start(runnable);
    Poco::ThreadPool::defaultPool().joinAll();
    return 0;

当你在 gdb 中单步执行它时,你会给线程时间在实例变量超出范围之前做它的事情。

于 2013-06-28T06:24:20.770 回答