linux 网络编程 socket bind failed 问题解决

今天写一个简单的socket网络通讯的程序的时候,用ctrl+c结束server端程序之后。再次启动server出现了bind failed:the address already in use的错误。在网上查了一下以后找到了原因。在此记录一下。这个IBM的官网上说到了这一点:http://www.ibm.com/developerworks/cn/linux/l-sockpit/

具体介绍例如以下:

bind 普遍遭遇的问题是试图绑定一个已经在使用的port。该陷阱是或许没有活动的套接字存在,但仍然禁止绑定port(bind 返回EADDRINUSE)。它由 TCP 套接字状态 TIME_WAIT 引起。

该状态在套接字关闭后约保留 2 到 4 分钟。

在 TIME_WAIT 状态退出之后,套接字被删除,该地址才干被又一次绑定而不出问题。

等待 TIME_WAIT 结束可能是令人恼火的一件事。特别是假设您正在开发一个套接字server。就须要停止server来做一些修改,然后重新启动。幸运的是,有方法能够避开 TIME_WAIT 状态。能够给套接字应用 SO_REUSEADDR 套接字选项,以便port能够立即重用。

考虑清单 3 的样例。

在绑定地址之前。我以 SO_REUSEADDR 选项调用 setsockopt

为了同意地址重用,我设置整型參数(on)为 1 (不然,能够设为 0 来禁止地址重用)。

清单 3.使用 SO_REUSEADDR 套接字选项避免地址使用错误
int sock, ret, on;
struct sockaddr_in servaddr;
/* Create a new stream (TCP) socket */
sock = socket( AF_INET, SOCK_STREAM, 0 ):
/* Enable address reuse */
on = 1;
ret = setsockopt( sock, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on) );
/* Allow connections to port 8080 from any available interface */
memset( &servaddr, 0, sizeof(servaddr) );
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = htonl( INADDR_ANY );
servaddr.sin_port = htons( 45000 );
/* Bind to the address (interface/port) */
ret = bind( sock, (struct sockaddr *)&servaddr, sizeof(servaddr) );

在应用了 SO_REUSEADDR 选项之后,bind API 函数将同意地址的马上重用。

我依照上面的指示增加这段代码以后又出现了一个错误:setsockopt failed: Socket operation on non-socket。

然后又在网上查,Socket operation on non-socket 错误出现有两种情况:

      1.  建立socket:

                 if(listenfd= socket(AF_INET,SOCK_STREAM, 0)==-1){

                      perror("creating socket failed!");
                      exit(-1);
                 }

           会造成在bind时出现 Socket operation on non-socket错误

           正确的代码应该是:

                  if((listenfd = socket(AF_INET, SOCK_STREAM, 0)) == -1){
                        perror("creating socket failed!");
                        exit(-1);
                  }

      2.  accept时:

                 if(connfd=accept(listenfd,(struct sockaddr *)&client_addr,(socklen_t *)&sin_size)==-1){
                      perror("accept error!");
                      exit(-1);
                 }

           会造成在recv时出现  Socket operation on non-socket错误

           正确代码是:

                 if((connfd=accept(listenfd,(struct sockaddr *)&client_addr,(socklen_t *)&sin_size))==-1){
                         perror("accept error!");
                         exit(-1);
                 }

   出现Socket operation on non-socket 错误的原因是:

            if(listenfd= socket(AF_INET,SOCK_STREAM, 0)==-1)

            if(connfd=accept(listenfd,(struct sockaddr *)&client_addr,(socklen_t *)&sin_size)==-1)

  这两句中缺失了()造成的。

赋值符合优先级最低,导致listenfd和connfd在创建/连接成功是为0,不成功时为1

最后:

如 调用connect  socket accept  函数出错,能够 perror("socket"); perror("connect"); printf("%s\n",strerror(errno));都能打出详细的错误。

编译结果出现例如以下错误:undefined reference to 'pthread_create',应该在在编译中要加 -lpthread參数。

转载于:https://www.cnblogs.com/ljbguanli/p/7280118.html

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
当然,我可以给你一个简单的 Linux Socket 编程的示例代码。以下是一个基于 TCP 协议的客户端和服务器端的示例: 客户端代码(client.c): ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <arpa/inet.h> #define SERVER_IP "127.0.0.1" #define SERVER_PORT 12345 int main() { int sockfd; struct sockaddr_in server_addr; char buffer[1024]; // 创建套接字 sockfd = socket(AF_INET, SOCK_STREAM, 0); if (sockfd < 0) { perror("Socket creation failed"); exit(EXIT_FAILURE); } memset(&server_addr, 0, sizeof(server_addr)); // 设置服务器地址 server_addr.sin_family = AF_INET; server_addr.sin_port = htons(SERVER_PORT); if (inet_pton(AF_INET, SERVER_IP, &(server_addr.sin_addr)) <= 0) { perror("Invalid address/ Address not supported"); exit(EXIT_FAILURE); } // 连接到服务器 if (connect(sockfd, (struct sockaddr*)&server_addr, sizeof(server_addr)) < 0) { perror("Connection failed"); exit(EXIT_FAILURE); } printf("Connected to the server.\n"); // 发送消息给服务器 strcpy(buffer, "Hello, server!"); if (write(sockfd, buffer, strlen(buffer)) < 0) { perror("Write failed"); exit(EXIT_FAILURE); } printf("Message sent to the server: %s\n", buffer); // 从服务器接收响应 memset(buffer, 0, sizeof(buffer)); if (read(sockfd, buffer, sizeof(buffer)) < 0) { perror("Read failed"); exit(EXIT_FAILURE); } printf("Response from the server: %s\n", buffer); // 关闭套接字 close(sockfd); return 0; } ``` 服务器端代码(server.c): ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <arpa/inet.h> #define SERVER_PORT 12345 int main() { int sockfd, new_sockfd; struct sockaddr_in server_addr, client_addr; socklen_t client_addr_len; char buffer[1024]; // 创建套接字 sockfd = socket(AF_INET, SOCK_STREAM, 0); if (sockfd < 0) { perror("Socket creation failed"); exit(EXIT_FAILURE); } memset(&server_addr, 0, sizeof(server_addr)); // 设置服务器地址 server_addr.sin_family = AF_INET; server_addr.sin_addr.s_addr = htonl(INADDR_ANY); server_addr.sin_port = htons(SERVER_PORT); // 绑定套接字到指定的 IP 地址和端口号 if (bind(sockfd, (struct sockaddr*)&server_addr, sizeof(server_addr)) < 0) { perror("Binding failed"); exit(EXIT_FAILURE); } // 监听连接请求 if (listen(sockfd, 5) < 0) { perror("Listening failed"); exit(EXIT_FAILURE); } printf("Server listening on port %d.\n", SERVER_PORT); while (1) { // 接收客户端连接 client_addr_len = sizeof(client_addr); new_sockfd = accept(sockfd, (struct sockaddr*)&client_addr, &client_addr_len); if (new_sockfd < 0) { perror("Accepting failed"); exit(EXIT_FAILURE); } printf("Client connected.\n"); // 接收客户端发送的消息 memset(buffer, 0, sizeof(buffer)); if (read(new_sockfd, buffer, sizeof(buffer)) < 0) { perror("Read failed"); exit(EXIT_FAILURE); } printf("Message from the client: %s\n", buffer); // 发送响应给客户端 strcpy(buffer, "Hello, client!"); if (write(new_sockfd, buffer, strlen(buffer)) < 0) { perror("Write failed"); exit(EXIT_FAILURE); } printf("Response sent to the client: %s\n", buffer); // 关闭新套接字 close(new_sockfd); printf("Client disconnected.\n"); } // 关闭套接字 close(sockfd); return 0; } ``` 请注意,这只是一个简单的示例代码,可能还需要进行错误处理和其他相关的细节处理。这个示例展示了一个最基本的 TCP 客户端和服务器端之间的通信过程。你可以根据你的需求进行修改和扩展。希望能对你有所帮助!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值