文章目录
概要
当第一次运行服务端时正常,第二次运行时出现Bind failed问题。
在实际使用中,如果您尝试启动一个服务端程序并且遇到了 "Bind failed" 的错误信息,这通常意味着尝试绑定(bind)的端口已经被占用。这种情况可能有以下几个常见原因:
-
端口占用:如果之前启动的服务端实例没有正确关闭,或者其他程序正在使用相同的端口,那么操作系统将不允许您再次绑定到该端口。
-
未正确释放端口:在服务端程序异常退出或者没有正确关闭套接字的情况下,操作系统可能仍然认为端口正在被使用。
-
TIME_WAIT状态:即使服务端正确关闭了套接字,TCP端口可能会因为TIME_WAIT状态而暂时不可用。TIME_WAIT是TCP连接正常关闭后的一个保留状态,用于确保所有的数据包都正确地完成传输。这个状态通常会持续一段时间(通常是几分钟)。
-
系统安全策略:某些操作系统可能有安全策略限制了对低端口号(通常是1024以下)的访问。这通常会要求服务端程序以管理员或特权用户身份运行。
为了解决 "Bind failed" 的问题,您可以采取以下几个步骤:
-
检查端口使用情况:在Unix-like系统中,可以使用 lsof -i :端口号 或 netstat -tulnp | grep 端口号 命令来检查端口占用情况。在Windows系统中,可以使用 netstat -aon | findstr 端口号。
-
确保正确关闭套接字:在您的代码中,确保在服务端关闭时,您调用了套接字的关闭方法。
-
设置SO_REUSEADDR套接字选项:这个选项允许重新绑定到正在TIME_WAIT状态的端口。在C/C++中,这可以通过设置套接字选项来完成:
cpp
int yes = 1;
if (setsockopt(server_socket, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(yes)) == -1) {
// 错误处理
-
更改端口号:尝试使用另一个未被占用的端口号。
-
检查权限:确保您的程序有足够的权限来绑定所需的端口。
-
重启系统:如果以上方法都无法解决问题,重启系统可以清除所有占用的端口,并重置网络堆栈。
样例代码
头文件
cpp
#include <ros/ros.h>
#include "std_msgs/String.h"
#include "std_msgs/Bool.h"
#include <iostream>
#include <cstring>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include "geometry_msgs/Twist.h"
#include <thread>
using namespace std;
class TCPPublisher
{
public:
TCPPublisher();
~TCPPublisher();
void cmdVelCallback(const geometry_msgs::Twist::ConstPtr& msg);
void acceptClients();
void sendToAllClients(const std::string& message);
private:
ros::NodeHandle n, nPrivate;
ros::Publisher tcpPub;
ros::Subscriber cmdVelSub;
// 创建服务器套接字
int serverSocket;
// 设置服务器地址结构
sockaddr_in serverAddr;
std::vector<int> clientSockets;
std::mutex clientSocketsMutex;
int clientSocket;
std::string topicStatus;
std::thread acceptThread;
};
main.cpp
cpp
#include "./tcp_pub/tcp_pub.h"
TCPPublisher::TCPPublisher():nPrivate("~")
{
nPrivate.param("topicStatus", topicStatus, std::string("/cmd_vel"));
// 创建套接字
serverSocket = socket(AF_INET, SOCK_STREAM, 0);
// 设置套接字选项,允许重新使用本地地址和端口
int yes = 1;
if (setsockopt(serverSocket, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(yes)) == -1) {
std::cerr << "Setsockopt failed." << std::endl;
close(serverSocket);
return;
}
/*订阅话题*/
cmdVelSub = n.subscribe(topicStatus.c_str(), 10, &TCPPublisher::cmdVelCallback, this);
// 设置服务器地址结构
sockaddr_in serverAddr;
serverAddr.sin_family = AF_INET;
serverAddr.sin_addr.s_addr = INADDR_ANY;
serverAddr.sin_port = htons(8080); // 服务器监听的端口号
// 绑定套接字
if (bind(serverSocket, (struct sockaddr*)&serverAddr, sizeof(serverAddr)) == -1) {
std::cerr << "Bind failed." << std::endl;
close(serverSocket);
return;
}
// 监听连接
if (listen(serverSocket, SOMAXCONN) == -1) {
std::cerr << "Listen failed." << std::endl;
close(serverSocket);
return;
}
std::cout << "Server is listening for incoming connections..." << std::endl;
// 在新线程中接受客户端连接
acceptThread = std::thread(&TCPPublisher::acceptClients, this);
ROS_INFO("TCPPublisher init successfully!!!");
}
TCPPublisher::~TCPPublisher()
{
close(serverSocket);
}
void TCPPublisher::cmdVelCallback(const geometry_msgs::Twist::ConstPtr& msg)
{
float velX = msg->linear.x;
float angularZ = msg->angular.z;
// ROS_INFO("velX : %f, angularZ : %f", velX, angularZ);
// 定义字符数组,用于存储转换后的结果
char buffer[50]; // 适当调整数组大小以适应你的需求
snprintf(buffer, sizeof(buffer), "%f,%f", velX, angularZ);
// ROS_INFO("buffer %s", buffer);
std::ostringstream ss;
ss << velX << "," << angularZ;
sendToAllClients(ss.str());
}
void TCPPublisher::acceptClients() {
while (ros::ok()) {
int clientSocket = accept(serverSocket, NULL, NULL);
if (clientSocket == -1) {
std::cerr << "Accept failed." << std::endl;
continue;
}
std::cout << "Connection established with a client." << std::endl;
// 添加到客户端列表
std::lock_guard<std::mutex> guard(clientSocketsMutex);
clientSockets.push_back(clientSocket);
}
}
void TCPPublisher::sendToAllClients(const std::string& message) {
std::lock_guard<std::mutex> guard(clientSocketsMutex);
for (auto it = clientSockets.begin(); it != clientSockets.end(); ) {
if (send(*it, message.c_str(), message.size(), 0) == -1) {
std::cerr << "Error sending message to client." << std::endl;
close(*it);
it = clientSockets.erase(it); // Remove from list if send fails
} else {
++it;
}
}
}
int main(int argc, char **argv) {
//创建节点
ros::init(argc, argv, "pure_pursuit");
TCPPublisher tp;
ros::spin();
return 0;
}