深入剖析 Linux 的 recv 函数原理与用途 (linux recv)

在 Linux 中,网络通信是十分重要的一部分。无论是客户端还是服务端,我们都离不开网络编程,而网络编程中最为核心的一部分便是 socket 编程。而在 socket 编程中,recv 函数便是一个非常重要的函数,它能够实现套接字接收数据,并填充缓存数据。本文将,带你了解这一函数的具体实现。

一、recv 函数的基本介绍

recv 函数是 Linux 中接收数据的一个函数,其函数原型为:

“`c

ssize_t recv(int sockfd, void *buf, size_t len, int flags);

“`

其中,sockfd 是指定一个待接收数据的目标套接字,buf 是指向缓冲区的指针,用于存储接收到的数据,len 表示缓冲区的大小,而 flags 则用于指定一些选项和标志。

二、recv 函数的返回值和错误码

recv 函数的返回值是一个 ssize_t 类型的整数,即它返回的接收到的字节数。如果在接收过程中出现了错误,recv 函数将会返回一个特定的错误码并设置 errno 变量。以下列出了一些常见的错误码:

| 错误码 | 描述 |

| —— | —————————————— |

| EAGN | 接收缓冲区中没有数据时会出现此错误 |

| EBADF | sockfd不是有效的文件描述符 |

| EINTR | 调用被信号中断 |

| EINVAL | 提供的缓冲区长度 len 是不合法的 |

| ENOMEM | 无法分配足够的内存来存储接收到的数据的缓冲 |

三、recv 函数的执行流程

recv 函数在执行时,首先会进行若干个判断,判断一些套接字和缓冲区是否合法,如缓冲区的长度是否为零,以及 sockfd 套接字是否有效等,如果发现不合法,就会返回错误码并设置 errno 变量。

在合法性判断完成后,recv 函数会阻塞等待数据,直到读取到指定长度 len 的数据或者遇到一些错误情况,recv 函数才会返回。如果缓冲区不足以存储全部接收到的数据(一般情况下,接收到数据长度将会是 len),那么该函数会存储 buffer 中的一部分,尽力而为地扩展缓存区,以便存储更多数据。

当阻塞等待数据的过程中出现某些异常情况时,recv 函数将会立即返回。这些异常情况包括:

1. 缓冲区中没有数据可供接收或者 sockfd 已经在非阻塞模式下接收过,此时 errno 的值将会被设置为 EAGN。

2. sockfd 连接已经被重置或者有丢包、误码等问题,此时 errno 的值将会被设置为 ECONNRESET。

3. sockfd 套接字被关闭,此时 errno 的值将会被设置为 ENOTCONN。

4. 连接对象错误或者权限问题,此时 errno 的值将会被设置为 EPERM。

四、recv 函数的使用场景

在实际开发中,recv 函数通常会被用于服务器端的网络程序中,以接收来自客户端的数据。当程序需要处理多个客户端请求时,recv 函数可以帮助服务器端处理来自多个客户端的请求,实现多任务并发处理。

此外,recv 函数还可以被用于实现数据的缓存,在recv 函数的实现中,允许缓冲器可以分配大于 len 个字节的缓冲区,这就可以用于实现数据的缓存。通过不断调用 recv 函数,不断接收新的数据,便可以实现数据的累加,实现数据的预处理,提高程序的性能。

这篇文章主要是介绍了 Linux 中 recv 函数的原理和用法,可以帮助大家更好地了解这一十分重要的函数。它可以用于服务器端接收数据、实现数据的缓存,以及提高程序性能等。当然,在实际开发中,我们还需要根据具体的情况,合理选用合适的函数来实现我们需要的功能。

相关问题拓展阅读:

Linux网络 – 数据包在内核中接收和发送的过程(转)

本文将介绍在Linux系统中,

数据包是如何一步一步从网卡传到进程手中的

以及

数据包是如何一步一步从应用程序到网卡并最终发送出去的

如果英文没有问题,强烈建议阅知弯读后面参考里的文章,里面介绍的更详细。

本文只讨论以太网的物理网卡,不涉及圆陆虚拟设备,并且以一个UDP包的接收过程作为示例.

网卡需要有驱动才能工作,驱动是加载到内核中的模块,负责衔接网卡和内核的网络模块,驱动在加载的时候将自己注册进网络模块,当相应的网卡收到数据包时,网络模块会调用相应的驱动程序处理数据。

下图展示了数据包(packet)如何进入内存,并被内核的网络模块开始处理:

软中断会触发内核网络模块中的软中断处理函数,后续流程如下

由于是UDP包,所以之一步会进入IP层,然后一级一级的函数往下调:

应用层一般有两种方式接收数据,一种是recvfrom函数阻塞在那里等着数据来,这种情况下当socket收到通知后,recvfrom就会被唤醒,然后读取接收队列的数据;另一种是通过epoll或者select监听相应的socket,当收到通知后,再调用recvfrom函数去读取接收队列的数据。两种情况都能正常的接收到相应的数据包。

了解数据包的接收流程有助于帮助我们搞清楚我们可以在哪些地方监控和修改数据包,哪些情况下数据包可能被丢弃,为我们处理网络问题提供了一些参考,同时了解netfilter中相应钩子的位置,对于了解iptables的用法有一定的帮助,同时也会帮助我们后续更好的理解Linux下的网络虚拟设备。

ndo_start_xmit会绑定到具体网卡驱动的相应函数,到这步之后,就归网卡驱动管了,不同的网卡驱动有不同的处理方式,搭腔闷这里不做详细介绍,其大概流程如下:

在网卡驱动发送数据包过程中,会有一些地方需要和netdevice子系统打交道,比如网卡的队列满了,需要告诉上层不要再发了,等队列有空闲的时候,再通知上层接着发数据。

linux recv的介绍就聊到这里吧,感谢你花时间阅读本站内容,更多关于linux recv,深入剖析 Linux 的 recv 函数原理与用途,Linux网络 – 数据包在内核中接收和发送的过程(转)的信息别忘了在本站进行查找喔。


数据运维技术 » 深入剖析 Linux 的 recv 函数原理与用途 (linux recv)