找回密码
 立即注册
首页 业界区 安全 TCP
煞赶峙 2025-6-11 16:17:33
TCP

TCP协议可靠性

TCP协议传输数据时会为每个字节分配一个序列号,通过这个序列号就可以判断数据是否重复到达以及数据是否丢失,还可以通过序列号对数据进行排序。另外,接收端每次接收到数据之后必须发送确认应答(ACK),如果在超时时间内没有发送应答信号(ACK),则TCP协议会认为数据没有送达,则会重新发送数据。
TCP协议判断数据是否损坏的方式是给每个数据段都添加校验和,接收端收到数据段进行校验,如果校验失败则丢弃已经损坏的数据段,也并不会进行确认应答,所以TCP协议会再次传输数据段。
因为通信可能是建立在不可靠的网络中以及网络层不可靠的传输机制,所以TCP协议中采用了一种*基于时钟的序列号握手机制*实现双方的有效连接,避免出现错误的连接初始化。
1.png

2.png

练习:主机A作为服务器,主机B作为客户端,主机A和主机B建立TCP连接,主机B可以向主机A发送消息
主机A
  1. #include <sys/socket.h>
  2. #include <netinet/in.h>
  3. #include
  4. #include <stdio.h>
  5. #include <errno.h>
  6. #include <sys/socket.h>
  7. #include <netinet/in.h>
  8. #include <netinet/ip.h>
  9. #include
  10. #include <sys/socket.h>
  11. #include <netinet/in.h>
  12. #include <netinet/udp.h>
  13. #include <stdlib.h>
  14. #include <string.h>
  15. #include <sys/types.h>
  16. #include <unistd.h>
  17. //TCP服务器代码   ./xxx   port
  18. int main(int argc, char const *argv[])
  19. {
  20.         //1.创建TCP套接字
  21.         int tcp_socket = socket(AF_INET, SOCK_STREAM, 0);
  22.         if (tcp_socket == -1)
  23.         {
  24.                 fprintf(stderr, "tcp socket error,errno:%d,%s\n",errno,strerror(errno));
  25.                 exit(1);
  26.         }
  27.         //2.绑定自身的IP地址和端口
  28.         struct sockaddr_in  host_addr;
  29.         host_addr.sin_family                 = AF_INET;                                                 //协议族,是固定的
  30.         host_addr.sin_port                   = htons(atoi(argv[1]));                        //目标端口,必须转换为网络字节序
  31.         host_addr.sin_addr.s_addr   = htonl(INADDR_ANY);                    //目标地址  INADDR_ANY 这个宏是一个整数,所以需要使用htonl转换为网络字节序
  32.         bind(tcp_socket,(struct sockaddr *)&host_addr, sizeof(host_addr));
  33.         //3.设置监听  队列最大容量是5
  34.         listen(tcp_socket,5);
  35.         //4.等待接受客户端的连接请求
  36.         struct sockaddr_in  client;
  37.         socklen_t client_len = sizeof(client);
  38.         int connect_fd = accept(tcp_socket,(struct sockaddr *)&client,&client_len); //会阻塞
  39.         char buf[128] = {0};
  40.         //5.说明双方建立连接,此时可以接收数据
  41.         while(1)
  42.         {
  43.                
  44.                
  45.                 read(connect_fd,buf,sizeof(buf));
  46.                 printf("recv from [%s],data is = %s\n", inet_ntoa(client.sin_addr) ,buf);
  47.                 bzero(buf,sizeof(buf));
  48.         }
  49.         return 0;
  50. }
复制代码
主机B(客户端)
  1. #include <sys/socket.h>
  2. #include <netinet/in.h>
  3. #include
  4. #include <stdio.h>
  5. #include <errno.h>
  6. #include <sys/socket.h>
  7. #include <netinet/in.h>
  8. #include <netinet/ip.h>
  9. #include
  10. #include <sys/socket.h>
  11. #include <netinet/in.h>
  12. #include <netinet/udp.h>
  13. #include <stdlib.h>
  14. #include <string.h>
  15. #include <sys/types.h>
  16. #include <unistd.h>
  17. //TCP客户端代码   ./xxx   port  IP
  18. int main(int argc, char const *argv[])
  19. {
  20.         //1.创建TCP套接字
  21.         int tcp_socket = socket(AF_INET, SOCK_STREAM, 0);
  22.         if (tcp_socket == -1)
  23.         {
  24.                 fprintf(stderr, "tcp socket error,errno:%d,%s\n",errno,strerror(errno));
  25.                 exit(1);
  26.         }
  27.        
  28.         //2.发起连接请求,等待接受服务器接受连接
  29.         struct sockaddr_in  dest_addr;
  30.         dest_addr.sin_family                 = AF_INET;                                                 //协议族,是固定的
  31.         dest_addr.sin_port                   = htons(atoi(argv[1]));                        //服务器端口,必须转换为网络字节序
  32.         dest_addr.sin_addr.s_addr   = inet_addr(argv[2]);                        //服务器地址 "192.168.64.xxx"  
  33.         int ret = connect(tcp_socket,(struct sockaddr *)&dest_addr,sizeof(dest_addr));
  34.         if (ret < 0)
  35.         {
  36.                 fprintf(stderr, "connect error,errno:%d,%s\n",errno,strerror(errno));
  37.                 exit(1);
  38.         }
  39.         char buf[128] = {0};
  40.         //3.说明双方建立连接,此时可以接收数据
  41.         while(1)
  42.         {
  43.                 printf("请输入:");
  44.                 scanf("%s",buf);
  45.                 write(tcp_socket,buf,sizeof(buf));
  46.                 bzero(buf,128);
  47.         }
  48.         return 0;
  49. }
复制代码
来源:程序园用户自行投稿发布,如果侵权,请联系站长删除
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
您需要登录后才可以回帖 登录 | 立即注册