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

Linux网络编程基础APITCP的数据读写API

《Linux高性能服务器编程》阅读笔记:1.TCP通信的读写(收发)接口因为”Linux系统上”一切皆是文件”,那么自然读写文件用的APIread()

《Linux高性能服务器编程》阅读笔记:


1. TCP通信的读写(收发)接口

  因为”Linux系统上”一切皆是文件”,那么自然读写文件用的API read()/write()同样适用于socket。不过Linux还提供了几个专门用于socket数据读写的系统调用。

#include
#include
ssize_t recv(int sockfd, void *buf, size_t len, int flags);
ssize_t send(int sockfd, const void *buf, size_t len, int flags);

  recv()用于读取sockfd上的数据,buf和len参数分别指定读缓冲区的位置和大小,flags参数用于数据收发的额外控制,通常设置为0。
  函数执行成功返回实际读取到的数据的长度,它可能小于程序员所期望的长度len。待读取的数据大于len时需要多次调用recv()才能完整读取,被读取过后的数据,将会在内核空间的接收缓冲区清除。recv()可能返回0,即对端已经关闭连接,出错则返回-1并设置errno。
  send()用于往sockfd写入数据,buf和len参数分别指定写缓冲区的位置和大小。send()成功时返回实际写入的数据的长度,失败则返回-1并设置errno。
  需要注意的是,send()实际是将应用程序的数据拷贝到内核的TCP模块的发送缓冲区,发往网络连接的具体实现是在系统内核。所以send()返回成功的情况下,也不能证明数据就一定发送成功
  flag参数的可选值有:
这里写图片描述

  flags参数只对send()/recv()的当前调用有效,要永久性的保持有效需要使用setsockopt()系统调用修改socket的属性。


2. 带外数据的收发

  带外数据即紧急数据,在TCP协议–带外数据和超时重传一文中有介绍。下面直接看代码。
  服务端(接收)代码:

#include
#include
#include
#include
#include
#include
#include
#include #define ERRP(con, ret, ...) do \
{ \if (con) \{ \perror(__VA_ARGS__); \ret; \} \
}while(0)#define BUFSIZE 1024
static char isstop = 0;int main(int argc, char** argv)
{if (argc <&#61; 2){printf("usage: %s ip_address port number backlog\n", basename(argv[0]));return -1;}const char* ip &#61; argv[1];int port &#61; atoi(argv[2]);int backlog &#61; 5;//1. 创建socketint socket_fd &#61; socket(AF_INET, SOCK_STREAM, 0);ERRP(socket_fd <&#61; 0, return -1, "socket");//2. 命名socketstruct 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 ret &#61; bind(socket_fd, (struct sockaddr* )&address, sizeof(address));ERRP(ret <0, return -1, "connect");//3. 为socket分配监听队列ret &#61; listen(socket_fd, backlog);ERRP(ret <0, return -1, "listen");//4. 从监听队列中取出连接struct sockaddr_in client;socklen_t client_addrlen &#61; sizeof(client);int connfd &#61; accept(socket_fd, (struct sockaddr* )&client, &client_addrlen);ERRP(connfd <0, return -1, "accept");printf("connect success: %s:%d\n", inet_ntoa(client.sin_addr), ntohs(client.sin_port));//5. 接收该连接的数据char buffer[BUFSIZE] &#61; {};memset(buffer, 0, BUFSIZE);ret &#61; recv(connfd, buffer, BUFSIZE - 1, 0);printf("got %d bytes of normal data: \"%s\"\n", ret, buffer); close(socket_fd);return 0;
}

  客户端(发送)代码&#xff1a;

int main(int argc, char** argv)
{if (argc <&#61; 2){printf("usage: %s ip_address port number backlog\n", basename(argv[0]));return -1;}const char* ip &#61; argv[1];int port &#61; atoi(argv[2]);int backlog &#61; 5;//1. 创建socketint socket_fd &#61; socket(AF_INET, SOCK_STREAM, 0);ERRP(socket_fd <&#61; 0, return -1, "socket");//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 ret &#61; connect(socket_fd, (struct sockaddr* )&address, sizeof(address));ERRP(ret <0, return -1, "connect");getchar();//3. 向服务端发送数据const char* oob_data &#61; "abcd";const char* normal_data &#61; "1234";send(socket_fd, normal_data, strlen(normal_data), 0);send(socket_fd, normal_data, strlen(normal_data), 0);//send(socket_fd, oob_data, strlen(oob_data), MSG_OOB); //MSG_OOB&#xff0c;发送的是带外数据send(socket_fd, normal_data, strlen(normal_data), 0);send(socket_fd, normal_data, strlen(normal_data), 0);getchar(); close(socket_fd);return 0;
}

  如上程序是发送正常数据的代码。运行结果&#xff1a;
这里写图片描述

  可见&#xff0c;即使发送端的数据分次发送&#xff0c;接收端也是一次性能读出。由于发送端没有发送带外数据&#xff0c;所以接收端以获取带外数据(MSG_OOB)时是失败返回的(-1)。接收端再次读取普通数据时会阻塞&#xff0c;因为接收缓冲区为空&#xff0c;阻塞直到发送端退出&#xff0c;此时接收端的阻塞读会返回0&#xff0c;表示对端已经退出。

  将发送端中向服务端发送数据代码片段的发送的是带外数据取消注释后&#xff0c;即向接收端发送带外数据&#xff0c;运行结果:
这里写图片描述

  可见&#xff0c;发送端发送的4字节的带外数据”abcd”仅最后一个字符’d’被接收端当成真正的带外数据接收&#xff0c;并且接收端对普通数据的接收会被带外数据截断。所以在带外数据接收之后还能接收到被截断后半部分的数据。

  我们还可以在这个通信过程中用tcpdump捕获双方通信的数据包&#xff1a;

$ sudo tcpdump -ntx -i lo port 9660

  跟数据的收发相关报文段如下&#xff1a;

(1) IP 192.168.239.136.44090 > 192.168.239.136.9660: Flags [P.], seq 1:5, ack 1, win 342, options [nop,nop,TS val 1538099 ecr 1535016], length 4
(2) IP 192.168.239.136.44090 > 192.168.239.136.9660: Flags [P.U], seq 5:13, ack 1, win 342, urg 8, options [nop,nop,TS val 1538099 ecr 1535016], length 8
(3) IP 192.168.239.136.44090 > 192.168.239.136.9660: Flags [P.], seq 13:17, ack 1, win 342, options [nop,nop,TS val 1538099 ecr 1535016], length 4
(3) IP 192.168.239.136.44090 > 192.168.239.136.9660: Flags [P.], seq 13:17, ack 1, win 342, options [nop,nop,TS val 1538099 ecr 1535016], length 4
(4) IP 192.168.239.136.44090 > 192.168.239.136.9660: Flags [P.], seq 17:21, ack 1, win 342, options [nop,nop,TS val 1538099 ecr 1535016], length 4

  (1) 报文段1是普通数据&#xff0c;即”1234”。
  (2) 报文段2是普通数据”1234”和带外数据”abcd”&#xff0c;这两份数据虽然是在发送端分两次发送&#xff0c;但是被内核优化成1次发送。注意&#xff0c;U标志表示该TCP报文段的头部被设置了紧急标志urg 8是紧急偏移值&#xff0c;它指出带外数据在完整字节流的位置是13(8 &#43; 5, 5是TCP报文段的序号值相对于初始序号值的偏移)&#xff0c;因此带外数据是完整字节流中的第12字节&#xff0c;即字符’d’。

  在实际应用中&#xff0c;接收端通常无法预期带外数据何时到来&#xff0c;而Linux内核在检测到TCP紧急标志时&#xff0c;可以有两种方式通知应用程序带外数据的到来: I/O复用产生的异常事件(如select()&#xff0c;其参数三exceptfds就是存放用于检测异常事件的描述符)和SIGURG信号&#xff0c;应有程序在知道有带外数据后&#xff0c;调用sockatmark()函数来定位到具体位置&#xff1a;

#include
int sockatmark(int sockfd);

  sockatmark()判断sockfd是否处于带外标记&#xff0c;即下一个被读取到的数据是否为带外数据&#xff0c;若不是则返回0&#xff1b;若是返回1&#xff0c;此时再利用带MSG_OOB的标志的recv()函数来接收带外数据。

  另外&#xff0c;补充一点&#xff0c;当对端close()通信连接时&#xff0c;若本端接着发送数据&#xff0c;根据TCP协议的规定&#xff0c;对端会收到一个RST复位报文&#xff0c;如果本端再往对端发送数据时&#xff0c;本端系统就会产生一个SIGPIPE信号。SIGPIPE信号默认会使得应用程序退出&#xff0c;程序员可以通过signal()系统调用设置该信号的具体行为。


推荐阅读
  • Nginx使用(server参数配置)
    本文介绍了Nginx的使用,重点讲解了server参数配置,包括端口号、主机名、根目录等内容。同时,还介绍了Nginx的反向代理功能。 ... [详细]
  • 本文介绍了为什么要使用多进程处理TCP服务端,多进程的好处包括可靠性高和处理大量数据时速度快。然而,多进程不能共享进程空间,因此有一些变量不能共享。文章还提供了使用多进程实现TCP服务端的代码,并对代码进行了详细注释。 ... [详细]
  • 本文介绍了Python高级网络编程及TCP/IP协议簇的OSI七层模型。首先简单介绍了七层模型的各层及其封装解封装过程。然后讨论了程序开发中涉及到的网络通信内容,主要包括TCP协议、UDP协议和IPV4协议。最后还介绍了socket编程、聊天socket实现、远程执行命令、上传文件、socketserver及其源码分析等相关内容。 ... [详细]
  • VScode格式化文档换行或不换行的设置方法
    本文介绍了在VScode中设置格式化文档换行或不换行的方法,包括使用插件和修改settings.json文件的内容。详细步骤为:找到settings.json文件,将其中的代码替换为指定的代码。 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • Python如何调用类里面的方法
    本文介绍了在Python中调用同一个类中的方法需要加上self参数,并且规范写法要求每个函数的第一个参数都为self。同时还介绍了如何调用另一个类中的方法。详细内容请阅读剩余部分。 ... [详细]
  • 本文介绍了基于c语言的mcs51单片机定时器计数器的应用教程,包括定时器的设置和计数方法,以及中断函数的使用。同时介绍了定时器应用的举例,包括定时器中断函数的编写和频率值的计算方法。主函数中设置了T0模式和T1计数的初值,并开启了T0和T1的中断,最后启动了CPU中断。 ... [详细]
  • Metasploit攻击渗透实践
    本文介绍了Metasploit攻击渗透实践的内容和要求,包括主动攻击、针对浏览器和客户端的攻击,以及成功应用辅助模块的实践过程。其中涉及使用Hydra在不知道密码的情况下攻击metsploit2靶机获取密码,以及攻击浏览器中的tomcat服务的具体步骤。同时还讲解了爆破密码的方法和设置攻击目标主机的相关参数。 ... [详细]
  • 本文介绍了使用PHP实现断点续传乱序合并文件的方法和源码。由于网络原因,文件需要分割成多个部分发送,因此无法按顺序接收。文章中提供了merge2.php的源码,通过使用shuffle函数打乱文件读取顺序,实现了乱序合并文件的功能。同时,还介绍了filesize、glob、unlink、fopen等相关函数的使用。阅读本文可以了解如何使用PHP实现断点续传乱序合并文件的具体步骤。 ... [详细]
  • 本文介绍了计算机网络的定义和通信流程,包括客户端编译文件、二进制转换、三层路由设备等。同时,还介绍了计算机网络中常用的关键词,如MAC地址和IP地址。 ... [详细]
  • 提升Python编程效率的十点建议
    本文介绍了提升Python编程效率的十点建议,包括不使用分号、选择合适的代码编辑器、遵循Python代码规范等。这些建议可以帮助开发者节省时间,提高编程效率。同时,还提供了相关参考链接供读者深入学习。 ... [详细]
  • 本文由编程笔记#小编为大家整理,主要介绍了logistic回归(线性和非线性)相关的知识,包括线性logistic回归的代码和数据集的分布情况。希望对你有一定的参考价值。 ... [详细]
  • 如何用UE4制作2D游戏文档——计算篇
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了如何用UE4制作2D游戏文档——计算篇相关的知识,希望对你有一定的参考价值。 ... [详细]
author-avatar
披着羊皮的狼19972010
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有