socket TCP_NODELAY选项

在元数据操作等小包传送时,发现性能不好,通过调试发现跟socket的TCP_NODELAY有很大关系。
TCP_NODELAY 和 TCP_CORK, 
这两个选项都对网络连接的行为具有重要的作用。许多UNIX系统都实现了TCP_NODELAY选项,但是,TCP_CORK则是Linux系统所独有的 而且相对较新;它首先在内核版本2.4上得以实现。此外,其他UNIX系统版本也有功能类似的选项,值得注意的是,在某种由BSD派生的系统上的 TCP_NOPUSH选项其实就是TCP_CORK的一部分具体实现。 
TCP_NODELAY和TCP_CORK基本上控制了包的“Nagle化”,Nagle化在这里的含义是采用Nagle算法把较小的包组装为更大的帧。 John Nagle是Nagle算法的发明人,后者就是用他的名字来命名的,他在1984年首次用这种方法来尝试解决福特汽车公司的网络拥塞问题(欲了解详情请参 看IETF RFC 896)。他解决的问题就是所谓的silly window syndrome ,中文称“愚蠢窗口症候群”,具体含义是,因为普遍终端应用程序每产生一次击键操作就会发送一个包,而典型情况下一个包会拥有一个字节的数据载荷以及40 个字节长的包头,于是产生4000%的过载,很轻易地就能令网络发生拥塞,。 Nagle化后来成了一种标准并且立即在因特网上得以实现。它现在已经成为缺省配置了,但在我们看来,有些场合下把这一选项关掉也是合乎需要的。 
现在让我们假设某个应用程序发出了一个请求,希望发送小块数据。我们可以选择立即发送数据或者等待产生更多的数据然后再一次发送两种策略。如果我们马上发 送数据,那么交互性的以及客户/服务器型的应用程序将极大地受益。例如,当我们正在发送一个较短的请求并且等候较大的响应时,相关过载与传输的数据总量相 比就会比较低,而且,如果请求立即发出那么响应时间也会快一些。以上操作可以通过设置套接字的TCP_NODELAY选项来完成,这样就禁用了Nagle 算法。 
另外一种情况则需要我们等到数据量达到最大时才通过网络一次发送全部数据,这种数据传输方式有益于大量数据的通信性能,典型的应用就是文件服务器。应用 Nagle算法在这种情况下就会产生问题。但是,如果你正在发送大量数据,你可以设置TCP_CORK选项禁用Nagle化,其方式正好同 TCP_NODELAY相反(TCP_CORK 和 TCP_NODELAY 是互相排斥的)。下面就让我们仔细分析下其工作原理。 
假设应用程序使用sendfile()函数来转移大量数据。应用协议通常要求发送某些信息来预先解释数据,这些信息其实就是报头内容。典型情况下报头很 小,而且套接字上设置了TCP_NODELAY。有报头的包将被立即传输,在某些情况下(取决于内部的包计数器),因为这个包成功地被对方收到后需要请求 对方确认。这样,大量数据的传输就会被推迟而且产生了不必要的网络流量交换。 
但是,如果我们在套接字上设置了TCP_CORK(可以比喻为在管道上插入“塞子”)选项,具有报头的包就会填补大量的数据,所有的数据都根据大小自动地 通过包传输出去。当数据传输完成时,最好取消TCP_CORK 选项设置给连接“拔去塞子”以便任一部分的帧都能发送出去。这同“塞住”网络连接同等重要。 
总而言之,如果你肯定能一起发送多个数据集合(例如HTTP响应的头和正文),那么我们建议你设置TCP_CORK选项,这样在这些数据之间不存在延迟。能极大地有益于WWW、FTP以及文件服务器的性能,同时也简化了你的工作。示例代码如下: 

intfd, on = 1; 
… 
/* 此处是创建套接字等操作,出于篇幅的考虑省略*/ 
… 
setsockopt (fd, SOL_TCP, TCP_CORK, &on, sizeof (on)); /* cork */ 
write (fd, …); 
fprintf (fd, …); 
sendfile (fd, …); 
write (fd, …); 
sendfile (fd, …); 
… 
on = 0; 
setsockopt (fd, SOL_TCP, TCP_CORK, &on, sizeof (on)); /* 拔去塞子 */ 
或者
setsockopt(s,IPPROTO_TCP,TCP_NODELAY,(char*)&yes,sizeof(int));
### 如何在使用 `curl` 时正确设置 TCP_NODELAY 选项 为了确保在网络应用中减少不必要的延迟,可以通过设置 `TCP_NODELAY` 来禁用 Nagle 算法。Nagle 算法旨在通过合并小的数据包来提高效率,但这可能会增加某些应用程序的响应时间。 对于 `libcurl` 库而言,可以直接利用其接口来启用或禁用此特性: ```c CURL *curl; struct curl_slist *headers = NULL; // 初始化 CURL 句柄 curl_global_init(CURL_GLOBAL_DEFAULT); curl = curl_easy_init(); if(curl) { // 启用 TCP_NODELAY (即关闭 Nagle 算法) long tcp_nodelay_option = 1L; // 1 表示开启, 0 表示关闭 curl_easy_setopt(curl, CURLOPT_TCP_FASTOPEN, tcp_nodelay_option); /* 执行其他必要的配置 */ // 进行请求... } ``` 需要注意的是,在上述代码片段中的 `CURLOPT_TCP_FASTOPEN` 实际上用于控制是否尝试使用 TCP Fast Open 而不是直接对应于 `TCP_NODELAY` 的设定;然而,`libcurl` 并未提供专门针对 `TCP_NODELAY` 的宏定义。因此,要真正实现这一点,则需借助底层 socket API 或者依赖特定平台上的扩展功能[^1]。 一种更通用的方法是在创建 HTTP 请求之前修改默认套接字行为,这通常涉及到操作系统层面的操作而不是单纯依靠 `curl` 提供的功能。例如,在 Linux 上可以在发起请求前执行如下命令以全局生效: ```bash echo "net.ipv4.tcp_timestamps=0" >> /etc/sysctl.conf echo "net.ipv4.tcp_tw_reuse=1" >> /etc/sysctl.conf sysctl -p ``` 不过这些更改会影响整个系统的网络性能,并不适合所有情况。更好的做法是仅对指定会话应用此类优化措施。为此目的,可以考虑编写自定义程序逻辑,在调用 `curl_easy_perform()` 前获取并调整当前连接对应的文件描述符属性。 另一种方式则是采用更高层次的语言绑定库(比如 Python 的 pycurl),它们可能提供了更加便捷的方式来处理这类需求。例如,在 Python 中可通过 PyCURL 设置 `TCP_NODELAY` 参数: ```python import pycurl from io import BytesIO buffer = BytesIO() crl = pycurl.Curl() # ... 配置 URL 和其他参数 ... # 开启 TCP_NODELAY crl.setopt(pycurl.TCP_NODELAY, True) try: crl.perform() finally: crl.close() ``` 这种方法允许开发者绕过标准 `curl` 工具链限制的同时保持较高的易用性和可移植性。 #### 解决相关问题建议 如果遇到无法成功设置 `TCP_NODELAY` 的情形,可能是由于权限不足或其他系统级约束所致。此时应检查是否有足够的权限去改变给定进程的工作模式,同时也应注意查看目标平台上是否存在额外的安全策略或防火墙规则阻碍了预期的行为发生。 此外,考虑到不同版本间可能存在差异,务必查阅官方文档确认所使用的软件发行版支持哪些特性和API接口。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值