一个端口号可以同时被两个进程绑定吗?( 三 )


SO_REUSEADDR允许单个进程捆绑同一端口到多个套接口上,只要每个捆绑指定不同的本地IP地址即可 。这一般不用于TCP服务器 。

  1. SO_REUSEADDR允许完全相同的地址和端口的重复绑定 。但这只用于UDP的多播,不用于TCP 。
SO_REUSEADDR允许完全重复的捆绑:当一个IP地址和端口绑定到某个套接口上时,还允许此IP地址和端口捆绑到另一个套接口上 。一般来说,这个特性仅在支持多播的系统上才有,而且只对UDP套接口而言(TCP不支持多播) 。
SO_REUSEPORT选项有如下语义: 此选项允许完全重复捆绑,但仅在想捆绑相同IP地址和端口的套接口都指定了此套接口选项才行 。
如果被捆绑的IP地址是一个多播地址,则SO_REUSEADDR和SO_REUSEPORT等效 。
使用这两个套接口选项的建议: 在所有TCP服务器中,在调用bind之前设置SO_REUSEADDR套接口选项; 当编写一个同一时刻在同一主机上可运行多次的多播应用程序时,设置SO_REUSEADDR选项,并将本组的多播地址作为本地IP地址捆绑 。
设置方法如下:
if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR,   (const void *)&nOptval , sizeof(int)) < 0)    ...
Q:编写 TCP/SOCK_STREAM 服务程序时,SO_REUSEADDR到底什么意思?
A:这个套接字选项通知内核,如果端口忙,但TCP状态位于 TIME_WAIT ,可以重用端口 。如果端口忙,而TCP状态位于其他状态,重用端口时依旧得到一个错误信息,指明"地址已经使用中" 。如果你的服务程序停止后想立即重启,而新套接字依旧使用同一端口,此时SO_REUSEADDR 选项非常有用 。必须意识到,此时任何非期望数据到达,都可能导致服务程序反应混乱,不过这只是一种可能,事实上很不可能 。
一个套接字由相关五元组构成,协议、本地地址、本地端口、远程地址、远程端口 。SO_REUSEADDR 仅仅表示可以重用本地本地地址、本地端口,整个相关五元组还是唯一确定的 。所以,重启后的服务程序有可能收到非期望数据 。必须慎重使用 SO_REUSEADDR 选项 。
举例例子1:测试上面第一种情况 。
#include <netinet/in.h> #include <sys/socket.h> #include <time.h> #include <stdio.h> #include <string.h> #define MAXLINE 100 int main(int argc, char** argv) {    int listenfd,connfd;    struct sockaddr_in servaddr;    char buff[MAXLINE+1];    time_t ticks;    unsigned short port;    int flag=1,len=sizeof(int);    port=10013;    if( (listenfd=socket(AF_INET,SOCK_STREAM,0)) == -1)    {      perror("socket");      exit(1);    }    bzero(&servaddr,sizeof(servaddr));    servaddr.sin_family=AF_INET;    servaddr.sin_addr.s_addr=htonl(INADDR_ANY);    servaddr.sin_port=htons(port);    if( setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &flag, len) == -1)   {       perror("setsockopt");       exit(1);    }    if( bind(listenfd,(struct sockaddr*)&servaddr,sizeof(servaddr)) == -1)    {       perror("bind");       exit(1);    }    else       printf("bind call OK!n");          if( listen(listenfd,5) == -1)    {       perror("listen");       exit(1);    }    for(;;)    {       if( (connfd=accept(listenfd,(struct sockaddr*)NULL,NULL)) == -1)      {           perror("accept");           exit(1);       }       if( fork() == 0)/*child process*/       {         close(listenfd);/*关闭监听套接字,子进程不需要 。*/                 ticks=time(NULL);         snprintf(buff,100,"%.24srn",ctime(&ticks));                 write(connfd,buff,strlen(buff));         close(connfd);                 sleep(1);                 execlp("run",NULL);         perror("execlp");         exit(1);      }      close(connfd);      exit(0);/* end parent*/   } } 


推荐阅读