C语言中经socket接收数据的相关函数详解

时间:2022-02-16 06:13:59

recv()函数:
头文件:

?
1
#include <sys/types.h>  #include <sys/socket.h>

定义函数:

?
1
int recv(int s, void *buf, int len, unsigned int flags);

函数说明:recv()用来接收远端主机经指定的socket 传来的数据, 并把数据存到由参数buf 指向的内存空间, 参数len 为可接收数据的最大长度.

参数 flags 一般设0. 其他数值定义如下:
1、MSG_OOB 接收以out-of-band 送出的数据.
2、MSG_PEEK 返回来的数据并不会在系统内删除, 如果再调用recv()会返回相同的数据内容.
3、MSG_WAITALL 强迫接收到len 大小的数据后才能返回, 除非有错误或信号产生.
4、MSG_NOSIGNAL 此操作不愿被SIGPIPE 信号中断返回值成功则返回接收到的字符数, 失败返回-1,错误原因存于errno 中.

错误代码:
   EBADF 参数s 非合法的socket 处理代码
   EFAULT 参数中有一指针指向无法存取的内存空间
   ENOTSOCK 参数s 为一文件描述词, 非socket.
   EINTR 被信号所中断
   EAGAIN 此动作会令进程阻断, 但参数s 的socket 为不可阻断
   ENOBUFS 系统的缓冲内存不足.
   ENOMEM 核心内存不足
   EINVAL 传给系统调用的参数不正确.

recvfrom()函数:
头文件:

?
1
#include <sys/types.h>  #include <sys/socket.h>

定义函数:

?
1
int recvfrom(int s, void *buf, int len, unsigned int flags, struct sockaddr *from,int *fromlen);

函数说明:recv()用来接收远程主机经指定的socket 传来的数据, 并把数据存到由参数buf 指向的内存空间, 参数len 为可接收数据的最大长度. 参数flags 一般设0, 其他数值定义请参考recv(). 参数from 用来指定欲传送的网络地址, 结构sockaddr 请参考bind(). 参数fromlen 为sockaddr 的结构长度.

返回值:成功则返回接收到的字符数, 失败则返回-1, 错误原因存于errno 中.

错误代码:
   EBADF 参数s 非合法的socket 处理代码
   EFAULT 参数中有一指针指向无法存取的内存空间.
   ENOTSOCK 参数s 为一文件描述词, 非socket.
   EINTR 被信号所中断.
   EAGAIN 此动作会令进程阻断, 但参数s 的socket 为不可阻断.
   ENOBUFS 系统的缓冲内存不足
   ENOMEM 核心内存不足
   EINVAL 传给系统调用的参数不正确.

范例:
/*利用socket 的UDP client此程序会连线UDP server, 并将键盘输入的字符串传给server.
UDP server 范例请参考sendto (). */

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/typs.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#define PORT 2345
#define SERVER_IP "127.0.0.1"
main()
{
  int s, len;
  struct sockaddr_in addr;
  int addr_len = sizeof(struct sockaddr_in);
  char buffer[256];
  //建立socket
  if((s = socket(AF_INET, SOCK_DGRAM, 0)) < 0)
  {
    perror("socket");
    exit(1);
  }
  //填写sockaddr_in
  bzero(&addr, sizeof(addr));
  addr.sin_family = AF_INET;
  addr.sin_port = htons(PORT);
  addr.sin_addr.s_addr = inet_addr(SERVER_IP);
  while(1)
  {
    bzero(buffer, sizeof(buffer));
    //从标准输入设备取得字符串
    len = read(STDIN_FILENO, buffer, sizeof(buffer));
    //将字符串传送给server 端
    sendto(s, buffer, len, 0, &addr, addr_len);
    //接收server 端返回的字符串
    len = recvfrom(s, buffer, sizeof(buffer), 0, &addr, &addr_len);
    printf("receive: %s", buffer);
  }
}

执行(先执行udp server 再执行udp client):

?
1
2
hello //从键盘输入字符串
receive: hello //server 端返回来的字符串

recvmsg()函数:
头文件:

?
1
#include <sys/types.h>  #include <sys/socktet.h>

定义函数:

?
1
int recvmsg(int s, struct msghdr *msg, unsigned int flags);

函数说明:recvmsg()用来接收远程主机经指定的socket 传来的数据. 参数s 为已建立好连线的socket, 如果利用UDP 协议则不需经过连线操作. 参数msg 指向欲连线的数据结构内容, 参数flags 一般设0, 详细描述请参考send(). 关于结构msghdr 的定义请参考sendmsg().

返回值:成功则返回接收到的字符数, 失败则返回-1, 错误原因存于errno 中.

错误代码:
   EBADF 参数s 非合法的socket 处理代码.
   EFAULT 参数中有一指针指向无法存取的内存空间
   ENOTSOCK 参数s 为一文件描述词, 非socket.
   EINTR 被信号所中断.
   EAGAIN 此操作会令进程阻断, 但参数s 的socket 为不可阻断.
   ENOBUFS 系统的缓冲内存不足
   ENOMEM 核心内存不足
   EINVAL 传给系统调用的参数不正确.