int ioctl(int d, int request, ...); 后续是否有参数根据第二个参数request的需要来决定。
ioctl用于硬件设备I/O通道控制,控制命令与参数都与设备高度相关,通常也与系统高度相关。
int fcntl(int fd, int cmd, ... /* arg */ );后续是否有参数根据第二个参数cmd的需要来决定。这点两者相同。
操作控制的对象是: 文件描述符。
ioctl()是底层的系统调用(system call),所以跨平台特性不好。
而fcntl则是被封装的函数,各个OS都是支持的。
在网络socket中,他们作用大概相似。
例如都可以设置socket的是否允许非阻塞模式,不过设置方式上略有不同:
int socketfd = .....;
fcntl(sockfd, F_SETFL,fcntl(sockfd, F_GETFL, 0) |O_NONBLOCK);
ioctl(sockfd, FIONBIO,1); 1:非阻塞0:阻塞
int setsockopt(int sockfd, int level, int optname, void *optval,socklen_t *optlen);
设置套接字选项.只能操作套接字。
sockfd: 套接字
level: 协议层 SOL_SOCKET/IPPROTO_IP/IPPRO_TCP
optname: 选项名 每一个协议层都有其固定的选项名
optval: 缓冲区 set是指向将要存放的地址, get是指向目前存放信息的地址
optlen: 缓冲区大小长度
在socket层常用操作有:
SO_BROADCAST 允许发送广播数据 int
SO_DEBUG 允许调试 int
SO_DONTROUTE 不查找路由 int
SO_ERROR 获得套接字错误 int
SO_KEEPALIVE 保持连接 int
SO_LINGER 延迟关闭连接 structlinger
SO_OOBINLINE 带外数据放入正常数据流 int
SO_RCVBUF 接收缓冲区大小 int
SO_SNDBUF 发送缓冲区大小 int
SO_RCVLOWAT 接收缓冲区下限 int
SO_SNDLOWAT 发送缓冲区下限 int
SO_RCVTIMEO 接收超时 structtimeval
SO_SNDTIMEO 发送超时 structtimeval
SO_REUSERADDR 允许重用本地地址和端口 int
SO_TYPE 获得套接字类型 int
SO_BSDCOMPAT 与BSD系统兼容 int
使用 SO_RECVBUF 和 SO_SENDBUF可以改变缺省缓冲区大小
例:
//设置接收缓冲区大小
intn_recvbuf = 32 * 1024;
setsockopt(sockfd, SOL_SOCKET,SO_RECVBUF, (const char*)&n_recvbuf,sizeof(int));
//设置发送缓冲区大小
intn_sendbuf = 32 * 1024;
setsockopt(sockfd, SOL_SOCKDET,SO_SENDBUF, (const char*)&n_sendbuf,sizeof(int));
使用SO_REUSERADDR:
1.当有一个有相同本地地址和端口的socket1处于TIME_WAIT状态时,而你启动的程序的socket2要占用该地址和端口,你的程 序就要用到该选项。
2.SO_RESUERADDR允许同一port上启动同一服务器的多个实例(多个进程),但每个实例绑定的IP地址是不能相同的。
3.SO_RESUERADDR允许单个进程绑定相同的端口到多个socket上,但每个socket绑定的ip地址不同。
4.SO_RESUERADDR允许完全相同的地址和端口的重复绑定,但这只用于UDP的多播,不用于TCP。
setsockopt()只是针对socket设置参数,是在连接中的参数控制,在OS层级的控制则由ioctl和fcntl控制。如果要是获取socket参数则使用getsockopt.
---------------------
作者:Richard__Lee
来源:CSDN
原文:https://blog.csdn.net/td_sexandzen/article/details/53785835
版权声明:本文为博主原创文章,转载请附上博文链接!