项目在使用websocket++这个开源库,作为websocket的服务器端来跟网页通信,虽然可以使用,但是有个问题就是在linux下的服务每次在关闭后马上启动,就会报端口重用的错误:
[2020-07-22 11:10:13] [info] asio listen error: system:98 (地址已在使用)
如果把服务停掉等待一分多钟后启动,就不会报这个错误了,说明端口已经被释放。如果监听的套接字设置套接字选项SO_REUSEADDR,就能解决,一般而言,作为服务器监听的套接字都必须设置这个套接字选项,因为这样才能保证服务在崩溃重启之后不会出现端口占用问题。
websocket++我使用的是boost的asio作为网络通信的库,可以说websocket++封装了boost asio的东西。刚开始没时间研究websocket++这个库的源码,都是在网上百度了一下别人的示例教程就自己封装了一下,主管又急着要项目上线,能用就用,这个重启端口重用问题算是小问题,既然那么说我也赖得管这些bug,然后就用了一段时间。现在有点时间要尝试自己解决一下。
自己看了一下这个服务器类
typedef websocketpp::server<websocketpp::config::asio> server;
看有没有提供什么接口来对底层的套接字选项进行设置。看了看websocketpp的源码,listen函数有如下源码:
void listen(lib::asio::ip::tcp::endpoint const & ep, lib::error_code & ec)
{
if (m_state != READY) {
m_elog->write(log::elevel::library,
"asio::listen called from the wrong state");
using websocketpp::error::make_error_code;
ec = make_error_code(websocketpp::error::invalid_state);
return;
}
m_alog->write(log::alevel::devel,"asio::listen");
lib::asio::error_code bec;
m_acceptor->open(ep.protocol(),bec);
if (bec) {ec = clean_up_listen_after_error(bec);return;}
m_acceptor->set_option(lib::asio::socket_base::reuse_address(m_reuse_addr),bec);
if (bec) {ec = clean_up_listen_after_error(bec);return;}
// if a TCP pre-bind handler is present, run it
if (m_tcp_pre_bind_handler) {
ec = m_tcp_pre_bind_handler(m_acceptor);
if (ec) {
ec = clean_up_listen_after_error(ec);
return;
}
}
m_acceptor->bind(ep,bec);
if (bec) {ec = clean_up_listen_after_error(bec);return;}
m_acceptor->listen(m_listen_backlog,bec);
if (bec) {ec = clean_up_listen_after_error(bec);return;}
// Success
m_state = LISTENING;
ec = lib::error_code();
}
这句 m_acceptor->set_option(lib::asio::socket_base::reuse_address(m_reuse_addr),bec);,就是设置套接字选项SO_REUSEADDR的,m_reuse_addr这是websocketpp::server<websocketpp::config::asio>类的成员变量,找了一下,这个类有个set_reuse_addr这个函数。大功告成,八九不离十,
。于是在套接字监听前加上这条语句(加粗那句):
//这里是我定义的全局服务器对象
//typedef websocketpp::server<websocketpp::config::asio> server;
//static server g_server;//全局的websocket服务器
g_server.set_reuse_addr(true);
g_server.listen(websocketpp::lib::asio::ip::tcp::v4(), uPort);
再编译重试了一遍,结果重启后就不会报端口重用的错误了,完美解决。
这里贴一下解决的问题的方法,希望遇到这个坑的童鞋可以找到这篇文章解决问题。