多种I/O函数(数据传输扩展)

前言:之前我们讲的数据传输一般Linux上用write和read,Windows上用send和recv。其实Linux上也可以用send和recv,它与write和read主要区别是它的最后一个参数可以附带一些扩展功能。

Linux中的send和recv基础

ssize_t send(int sockfd, const void *buf, size_t nbytes, int flags); 成功返回发送的字节数,失败返回-1 参数: sockfd:套接字文件描述符 buf:保存传输数据的缓冲地址值 nbytes:传输的字节数 flags:扩展信息

ssize_t recv(int sockfd, void *buf, size_t nbytes, int flags); 成功返回接收的字节数(收到EOF返回0),失败返回-1 参数: sockfd:套接字文件描述符 buf:保存接收数据的缓冲地址值 nbytes:可接收的最大字节数 flags:扩展信息

这两个函数主要讲的就是最后一个参数flags的扩展信息,,以前我们都是没有使用它直接传的0,这些扩展信息可选项可以利用位或运算(|)同时传递多个信息。可选项如下:

MSG_OOB:传输紧急消息(Out-of-band data) MSG_PEEK:验证输入缓冲中是否存在接收的数据 MSG_DONTROUTE:在本地网络中寻找目的地 MSG_DONTWAIT:非阻塞I/O MSG_WAITALL:防止函数返回,直到接收全部请求的字节数 注意:不同操作系统对上述可选项的支持也不同。

代码示例(MSG_OOB为例) 服务端:error_handling(char *message);void urg_handler(int signo);int acpt_sock;int recv_sock;int main(int argc, const char * argv[]) {struct sockaddr_in recv_adr, serv_adr;int str_len, state;socklen_t serv_adr_sz;struct sigaction act;char buf[BUF_SIZE];if (argc != 2) {printf(“Usage: %s <port> \n”, argv[0]);exit(1);}//Linux上的信号处理(事件驱动),Windows可以用select函数模拟act.sa_handler = urg_handler; //回调函数sigemptyset(&act.sa_mask); //初始化0act.sa_flags = 0;acpt_sock = socket(PF_INET, SOCK_STREAM, 0);memset(&recv_adr, 0, sizeof(recv_adr));recv_adr.sin_family = AF_INET;recv_adr.sin_addr.s_addr = htonl(INADDR_ANY);recv_adr.sin_port = htons(atoi(argv[1]));if(bind(acpt_sock, (struct sockaddr *) &recv_adr, sizeof(recv_adr)) == -1)error_handling(“bind() error”);if(listen(acpt_sock, 5) == -1)error_handling(“listen() error”);serv_adr_sz = sizeof(serv_adr);recv_sock = accept(acpt_sock, (struct sockaddr *)&serv_adr, &serv_adr_sz);//将引发信号事件的句柄recv_sock改为getpid()生成的ID,防止多进程中子进程也响应这个事件fcntl(recv_sock, F_SETOWN, getpid());state = sigaction(SIGURG, &act, ((str_len = recv(recv_sock, buf, sizeof(buf), 0)) != 0){if (str_len == -1)continue;buf[str_len] = 0;puts(buf);}close(recv_sock);close(acpt_sock);return 0;}//紧急消息接收void urg_handler(int signo){int str_len;char buf[BUF_SIZE];str_len = recv(recv_sock, buf, sizeof(buf) – 1, MSG_OOB);buf[str_len] = 0;printf(“Urgent message : %s \n”, buf);}void error_handling(char *message){fputs(message, stderr);fputc(‘\n’, stderr);exit(1);}

客服端:

error_handling(char *message);int main(int argc, const char * argv[]) {int sock;struct sockaddr_in recv_adr;if(argc != 3){printf(“Usage: %s <IP> <port> \n”, argv[0]);exit(1);}sock = socket(PF_INET, SOCK_STREAM, 0);if(sock == -1)error_handling(“socket() error”);memset(&recv_adr, 0, sizeof(recv_adr));recv_adr.sin_family = AF_INET;recv_adr.sin_addr.s_addr = inet_addr(argv[1]);recv_adr.sin_port = htons(atoi(argv[2]));if (connect(sock, (struct sockaddr *) &recv_adr, sizeof(recv_adr)) == -1)error_handling(“connect() error”);//发送紧急消息,Lunix上是信号处理(事件驱动),Windows上可以select函数模拟write(sock, “123”, strlen(“123”));send(sock, “4”, strlen(“4”), MSG_OOB);sleep(2); //os上紧急消息同时发送,下一条会替换上一条,同一时间只能保存一条(可能是一个变量保存的,不是缓冲数组)write(sock, “567”, strlen(“567”));send(sock, “890”, strlen(“890”), MSG_OOB);close(sock);return 0;}void error_handling(char *message){fputs(message, stderr);fputc(‘\n’, stderr);exit(1);}

readv和writev函数

基础 这两个函数有助于提高数据通信效率,它们能对数据进行整合传输及发送,适当使用这2个函数可以减少I/O函数的调用次数。

阳光总在风雨后。只有坚强的忍耐顽强的奋斗,

多种I/O函数(数据传输扩展)

相关文章:

你感兴趣的文章:

标签云: