热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

开发笔记:重要的socket选项

篇首语:本文由编程笔记#小编为大家整理,主要介绍了重要的socket选项相关的知识,希望对你有一定的参考价值。

篇首语:本文由编程笔记#小编为大家整理,主要介绍了重要的socket选项相关的知识,希望对你有一定的参考价值。






下面两个系统调用专门用来读取和设置socket文件描述符属性的方法:

int getsockopt(int sockfd, int level, int option_name, void* option_value, socklen_t* restrict option_len);
int setsockopt(int sockfd, int level, int option_name, const void* option_value, socklen_t restrict option_len);

level指定要操作那个协议的选项(即属性),比如IPV4、IIPV6、TCP等
option_name指定选项的名字
option_valueoption_len指定被操作的值和长度
在这里插入图片描述

值得指出的是,对服务器而言,有部分socket选项只能在调用listen系统调用前针对监听socket设置才有效。这是因为连接socket只能由accept调用返回,而acceptlisten监听队列中接受的连接至少已经完成了TCP三次握手的前两个步骤(因为listen监听队列中的连接至少已进入SYN_RCVD状态),这说明服务器已经往被接受连接上发送了TCP同步报文段。但有的socket选项却应该在TCP同步报文段(置SYN标志位的报文段)中设置,比如TCP最大报文段选项。对这种情况,解决方案是:对监听socket设置这些socket选项,那么accept返回的连接socket将自动继承这些选项。


SO_REUSEADDR选项

服务器程序可以通过设置socket选项SO_REUSEADDR来强制使用被处于TIME_WAIT状态的连接占用socket地址


SO_RCVBUF和SO_SNDBUF选项

SO_RCVBUFSO_SNDBUF分别表示接受缓冲区和发送缓冲区的大小。
不过当我们用setsockopt来设置TCP的接受缓冲区的发送缓冲区时,系统都会将其值加倍,并且不得小于某个系统默认的最小值。因为系统要确保一个TCP连接拥有足够的空闲缓冲区来处理拥塞(比如快速重传算法就希望TCP接收缓冲区能够容纳4个大小为SMSS的TCP报文段)
我们编写一对客户端和服务端程序,分别修改TCP发送缓冲区和接受缓冲区的大小:
修改TCP发送缓冲区的客户端程序:set_send_buf.c

#include
#include
#include
#include
#include
#include
#include
#include
#define BUFFER_SIZE 512
int main(int argc, char* argv[]) {
if (argc <&#61; 2) {
printf("usage: %s ip_address port_number send_buffer_size", basename(argv[0]));
return 1;
}
const char* ip &#61; argv[1];
int port &#61; atoi(argv[2]);
struct sockaddr_in server_address;
bzero(&server_address, sizeof(server_address));
server_address.sin_family &#61; AF_INET;
inet_pton(AF_INET, ip, &server_address.sin_addr);
server_address.sin_port &#61; htons(port);
int sock &#61; socket(PF_INET, SOCK_STREAM, 0);
assert(sock >&#61; 0);
int sendbuf &#61; atoi(argv[3]);
int len &#61; sizeof(sendbuf);
// 先设置TCP发送缓冲区的大小&#xff0c;然后立即读取之
setsockopt(sock, SOL_SOCKET, SO_SNDBUF, &sendbuf, sizeof(sendbuf));
// getsockopt(sock, SOL_SOCKET, SO_SNDBUF, &sendbuf, (struct socklen*)&len);
getsockopt(sock, SOL_SOCKET, SO_SNDBUF, &sendbuf, (socklen_t*)&len);
printf("the tcp send buffer size after setting is %d\\n", sendbuf);

if (connect(sock, (struct sockaddr*)&server_address, sizeof(server_address)) !&#61; -1) {
char buffer[BUFFER_SIZE];
memset(&buffer, &#39;\\0&#39;, BUFFER_SIZE);
send(sock, &buffer, BUFFER_SIZE, 0);
}
close(sock);
return 0;
}

修改TCP接受缓冲区的服务端程序&#xff1a;set_recv_buf.c

#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define BUFFER_SIZE 1024
#define BACKLOG 5
int main(int argc, char* argv[]) {
if (argc <&#61; 2) {
printf("usage: %s ip_address port_number recv_buffer_size", basename(argv[0]));
return 1;
}
const char* ip &#61; argv[1];
int port &#61; atoi(argv[2]);
struct sockaddr_in address;
bzero(&address, sizeof(address));
address.sin_family &#61; AF_INET;
inet_pton(AF_INET, ip, &address.sin_addr);
address.sin_port &#61; htons(port);
int sockfd &#61; socket(PF_INET, SOCK_STREAM, 0);
assert(sockfd >&#61; 0);
int recvbuf &#61; atoi(argv[3]);
int len &#61; sizeof(recvbuf);
// 先设置TCP接受缓冲区的大小&#xff0c;然后立即读取
setsockopt(sockfd, SOL_SOCKET, SO_RCVBUF, &recvbuf, sizeof(recvbuf));
getsockopt(sockfd, SOL_SOCKET, SO_RCVBUF, &recvbuf, (socklen_t*)&len);
printf("the tcp recvive buffer size after setting is %d\\n", recvbuf);
int ret &#61; bind(sockfd, (struct sockaddr*)&address, sizeof(address));
assert(ret !&#61; -1);
ret &#61; listen(sockfd, BACKLOG);
assert(ret !&#61; -1);
struct sockaddr_in client;
socklen_t client_addrlength &#61; sizeof(client);
// int connfd &#61; accept(sockfd, &client, &client_addrlength);
int connfd &#61; accept(sockfd, (struct sockaddr*)&client, &client_addrlength);
if (connfd < 0) {
printf("errno is: %d\\n", errno);
} else {
char buffer[BUFFER_SIZE];
memset(buffer, &#39;\\0&#39;, sizeof(buffer));
while (recv(connfd, &buffer, BUFFER_SIZE - 1, 0) > 0) {}
close(connfd);
}
close(sockfd);
return 0;
}

我们将接收缓冲区大小设置为50&#xff0c;将发送缓冲区大小设置为2000&#xff0c;得到如下结果
在这里插入图片描述
在这里插入图片描述

他们都没有按照我所要求的50、2000设置&#xff0c;应该是因为系统当前接收缓冲区最小值为2304&#xff0c;发送缓冲区最小为4608.

接下来将接收缓冲区设置为1000、1152
在这里插入图片描述
在这里插入图片描述

依旧为2304
但如果设置为1153&#xff0c;则
在这里插入图片描述
缓冲区设置为我所要求的两倍&#xff0c;即2306。说明如果当我们的设置加倍后仍然小于系统要求的最小值&#xff0c;则会采用系统要求的最小值&#xff0c;否则为我所设置的乘2.


SO_RCVLOWAT和SO_SNDWAT选项

SO_RCVLOWATSO_SNDLOWAT选项分别表示TCP接收缓冲区和发送缓冲区的低水位标记。它们一般被IO复用系统调用用来判断socket是否可读或可写。当TCP接收缓冲区中可读数据的总数大于其低水位标记时&#xff0c;I/O复用系统调用将通知应用程序可以从对应的socket.上读取数据&#xff1b;当TCP发送缓冲区中的空闲空间&#xff08;可以写入数据的空间&#xff09;大于其低水位标记时&#xff0c;I/O 复用系统调用将通知应用程序可以往对应的socke。上写人数据。默认情况下&#xff0c;TCP接收缓冲区的低水位标记和TCP发送缓冲区的低水位标记均为1字节。


SO_LINGER选项

控制close系统调用在关闭TCP连接时的行为。






推荐阅读
  • 本文讨论了clone的fork与pthread_create创建线程的不同之处。进程是一个指令执行流及其执行环境,其执行环境是一个系统资源的集合。在调用系统调用fork创建一个进程时,子进程只是完全复制父进程的资源,这样得到的子进程独立于父进程,具有良好的并发性。但是二者之间的通讯需要通过专门的通讯机制,另外通过fork创建子进程系统开销很大。因此,在某些情况下,使用clone或pthread_create创建线程可能更加高效。 ... [详细]
  • 本文介绍了深入浅出Linux设备驱动编程的重要性,以及两种加载和删除Linux内核模块的方法。通过一个内核模块的例子,展示了模块的编译和加载过程,并讨论了模块对内核大小的控制。深入理解Linux设备驱动编程对于开发者来说非常重要。 ... [详细]
  • 本文详细介绍了GetModuleFileName函数的用法,该函数可以用于获取当前模块所在的路径,方便进行文件操作和读取配置信息。文章通过示例代码和详细的解释,帮助读者理解和使用该函数。同时,还提供了相关的API函数声明和说明。 ... [详细]
  • 本文介绍了设计师伊振华受邀参与沈阳市智慧城市运行管理中心项目的整体设计,并以数字赋能和创新驱动高质量发展的理念,建设了集成、智慧、高效的一体化城市综合管理平台,促进了城市的数字化转型。该中心被称为当代城市的智能心脏,为沈阳市的智慧城市建设做出了重要贡献。 ... [详细]
  • eclipse学习(第三章:ssh中的Hibernate)——11.Hibernate的缓存(2级缓存,get和load)
    本文介绍了eclipse学习中的第三章内容,主要讲解了ssh中的Hibernate的缓存,包括2级缓存和get方法、load方法的区别。文章还涉及了项目实践和相关知识点的讲解。 ... [详细]
  • Linux环境变量函数getenv、putenv、setenv和unsetenv详解
    本文详细解释了Linux中的环境变量函数getenv、putenv、setenv和unsetenv的用法和功能。通过使用这些函数,可以获取、设置和删除环境变量的值。同时给出了相应的函数原型、参数说明和返回值。通过示例代码演示了如何使用getenv函数获取环境变量的值,并打印出来。 ... [详细]
  • 海马s5近光灯能否直接更换为H7?
    本文主要介绍了海马s5车型的近光灯是否可以直接更换为H7灯泡,并提供了完整的教程下载地址。此外,还详细讲解了DSP功能函数中的数据拷贝、数据填充和浮点数转换为定点数的相关内容。 ... [详细]
  • 本文介绍了使用哈夫曼树实现文件压缩和解压的方法。首先对数据结构课程设计中的代码进行了分析,包括使用时间调用、常量定义和统计文件中各个字符时相关的结构体。然后讨论了哈夫曼树的实现原理和算法。最后介绍了文件压缩和解压的具体步骤,包括字符统计、构建哈夫曼树、生成编码表、编码和解码过程。通过实例演示了文件压缩和解压的效果。本文的内容对于理解哈夫曼树的实现原理和应用具有一定的参考价值。 ... [详细]
  • OpenMap教程4 – 图层概述
    本文介绍了OpenMap教程4中关于地图图层的内容,包括将ShapeLayer添加到MapBean中的方法,OpenMap支持的图层类型以及使用BufferedLayer创建图像的MapBean。此外,还介绍了Layer背景标志的作用和OMGraphicHandlerLayer的基础层类。 ... [详细]
  • 本文介绍了在PostgreSQL中批量导入数据时的优化方法。包括使用unlogged表、删除重建索引、删除重建外键、禁用触发器、使用COPY方法、批量插入等。同时还提到了一些参数优化的注意事项,如设置effective_cache_size、shared_buffer等,并强调了在导入大量数据后使用analyze命令重新收集统计信息的重要性。 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • PHP中的单例模式与静态变量的区别及使用方法
    本文介绍了PHP中的单例模式与静态变量的区别及使用方法。在PHP中,静态变量的存活周期仅仅是每次PHP的会话周期,与Java、C++不同。静态变量在PHP中的作用域仅限于当前文件内,在函数或类中可以传递变量。本文还通过示例代码解释了静态变量在函数和类中的使用方法,并说明了静态变量的生命周期与结构体的生命周期相关联。同时,本文还介绍了静态变量在类中的使用方法,并通过示例代码展示了如何在类中使用静态变量。 ... [详细]
  • 本文介绍了Swing组件的用法,重点讲解了图标接口的定义和创建方法。图标接口用来将图标与各种组件相关联,可以是简单的绘画或使用磁盘上的GIF格式图像。文章详细介绍了图标接口的属性和绘制方法,并给出了一个菱形图标的实现示例。该示例可以配置图标的尺寸、颜色和填充状态。 ... [详细]
  • 本文讨论了如何使用GStreamer来删除H264格式视频文件中的中间部分,而不需要进行重编码。作者提出了使用gst_element_seek(...)函数来实现这个目标的思路,并提到遇到了一个解决不了的BUG。文章还列举了8个解决方案,希望能够得到更好的思路。 ... [详细]
  • 流数据流和IO流的使用及应用
    本文介绍了流数据流和IO流的基本概念和用法,包括输入流、输出流、字节流、字符流、缓冲区等。同时还介绍了异常处理和常用的流类,如FileReader、FileWriter、FileInputStream、FileOutputStream、OutputStreamWriter、InputStreamReader、BufferedReader、BufferedWriter等。此外,还介绍了系统流和标准流的使用。 ... [详细]
author-avatar
婉婷雅铃43
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有