在Linux上“已接受” fd上的setsockopt
内容导读
互联网集市收集整理的这篇技术教程文章主要介绍了在Linux上“已接受” fd上的setsockopt,小编现在分享给大家,供广大互联网技能从业者学习和参考。文章包含2698字,纯文字阅读大概需要4分钟。
内容图文
对于用于SO_REUSEADDR的Linux,setsockopt的行为我有一个相当奇怪的发现.一行:如果我将sockopt应用于“侦听套接字”上accept所返回的fd上,则套接字选项将反映在侦听套接字所持有的端口上.
确定一些代码.
服务器:打开套接字,将SO_REUSEADDR设置为true.接受连接,然后在accept返回的fd上的fd上将SO_REUSEADDR设置为false.
#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <arpa/inet.h>
#include <string.h>
int main(void)
{
int s, len;
int sin_size;
int reuse = 1;
int ret;
struct sockaddr_in my_addr;
memset(&my_addr, 0, sizeof(my_addr));
my_addr.sin_family = AF_INET;
my_addr.sin_addr.s_addr = inet_addr("127.0.0.1");
my_addr.sin_port = htons(33235);
if( (s = socket(AF_INET, SOCK_STREAM, 0)) < 0)
{
printf("Socket Error\n");
return -1;
}
setsockopt(s, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(int));
if( bind(s, (struct sockaddr*)&my_addr, sizeof(struct sockaddr)) < 0)
{
printf("Bind Error\n");
return -1;
}
listen(s, 6);
reuse = 0;
memset(&my_addr, 0, sizeof(my_addr));
while(1) {
ret = accept(s, (struct sockaddr*)&my_addr, &len);
if (ret<0) {
printf("Accept failed\n");
} else {
printf("Accepted a client setting reuse add to 0\n");
setsockopt(ret, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(int));
}
}
printf("Server exiting\n");
return 0;
}
客户端:客户端连接到服务器,并且在此之后不执行任何操作,以确保服务器套接字保持TIME_WAIT状态.
#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <arpa/inet.h>
#include <string.h>
#include <errno.h>
int main(void)
{
int s, len;
int sin_size;
struct sockaddr_in my_addr;
memset(&my_addr, 0, sizeof(my_addr));
my_addr.sin_family = AF_INET;
my_addr.sin_addr.s_addr = inet_addr("127.0.0.1");
my_addr.sin_port = htons(33235);
if( (s = socket(AF_INET, SOCK_STREAM, 0)) < 0)
{
printf("Socket Error\n");
return -1;
}
if (!connect(s,(struct sockaddr*)&my_addr, sizeof(struct sockaddr)))
{
printf("Client Connected successfully\n");
}
else
{
printf("%s\n",strerror(errno));
}
while(1) sleep(1);
return 0;
}
我确实会重现此问题的步骤.
>运行服务器.
>连接客户端.
>杀死并重新启动服务器.服务器因绑定失败而失败
我在Mac OS上进行了测试.绑定没有失败.我已经挖掘了所有Posix规范,但没有一个说这个代码是未定义的.
题:
对此有更多经验的人可以分享对这个问题的理解吗?
解决方法:
考虑这种情况的一种方法是SO_REUSEADDR确定是否可以将另一个套接字绑定到同一地址.它是任何套接字(侦听或连接)的属性,但非常普遍地是从通过accept的监听继承的.在Linux中,它映射到struct sock“ sk_reuse”标志.
如果清除了您“接受”的FD上的此标志,则从该点开始IP /端口对将被视为忙碌且不可重用.侦听套接字上的SO_REUSEADDR标志不会更改,但是接受的套接字上的标志会影响绑定逻辑.您可能可以使用getsockopt进行检查.
如果您想了解更多,可以尝试阅读inet_csk_get_port函数:http://lxr.free-electrons.com/source/net/ipv4/inet_connection_sock.c#L100.这是实际的“绑定”发生的地方.
内容总结
以上是互联网集市为您收集整理的在Linux上“已接受” fd上的setsockopt全部内容,希望文章能够帮你解决在Linux上“已接受” fd上的setsockopt所遇到的程序开发问题。 如果觉得互联网集市技术教程内容还不错,欢迎将互联网集市网站推荐给程序员好友。
内容备注
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 gblab@vip.qq.com 举报,一经查实,本站将立刻删除。
内容手机端
扫描二维码推送至手机访问。