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 版权声明:本文为博主原创文章,转载请附上博文链接!