首页 > 编程语言 >学懂C++(三十九):网络编程——深入详解 TCP 和 UDP 的区别和应用场景

学懂C++(三十九):网络编程——深入详解 TCP 和 UDP 的区别和应用场景

时间:2024-08-20 12:52:14浏览次数:19  
标签:UDP addr C++ server 学懂 TCP include 数据包

目录

一、TCP 的特点及应用场景

1. 可靠性

2. 流控制和拥塞控制

3. 有序传输

4. 应用场景

二、UDP 的特点及应用场景

1. 无连接

2. 不可靠性

3. 轻量级

4. 支持广播和多播

5. 应用场景

三、TCP 和 UDP 的区别

四、TCP 和 UDP 的工作原理

1. TCP 的工作原理

三次握手

数据传输

四次挥手

2. UDP 的工作原理

数据传输

广播和多播

五、实例比较

TCP 示例

TCP 服务器

TCP 客户端

UDP 示例

UDP 服务器

UDP 客户端

六、总结


        网络通信中,传输控制协议 (TCP) 和用户数据报协议 (UDP) 是两种主要的传输层协议。它们在数据传输方式、可靠性、连接管理等方面有着显著的区别。本文将深入解析 TCP 和 UDP 的工作原理、主要特性及其应用场景,帮助读者更好地理解和选择合适的传输协议。

一、TCP 的特点及应用场景

传输控制协议 (TCP, Transmission Control Protocol) 是一种面向连接的、可靠的传输协议,适用于对数据传输可靠性要求较高的应用场景。以下是 TCP 的主要特点和应用场景:

1. 可靠性

TCP 提供可靠的数据传输,通过以下机制保证数据的完整性和顺序性:

  • 三次握手:在传输数据前,双方需要进行三次握手以建立连接,确保双方都准备好发送和接收数据。
  • 序列号和确认应答:每个数据包都有一个序列号,接收方会发送确认应答 (ACK) 来确认收到的数据。若发送方未收到 ACK,则会重传数据包。
  • 校验和:每个数据包附带校验和,以检测传输过程中是否出现数据错误。
  • 重传机制:如在规定时间内未收到 ACK,发送方会自动重传数据包,直到确认数据包成功到达。
2. 流控制和拥塞控制

TCP 实现了流控制和拥塞控制机制:

  • 滑动窗口:TCP 使用滑动窗口机制控制数据流量,防止发送方发送过多数据,导致接收方缓冲区溢出。
  • 拥塞控制:TCP 采用慢启动、拥塞避免、快速重传和快速恢复等算法,动态调整数据发送速率,避免网络拥塞。
3. 有序传输

TCP 保证数据包按序到达,即使在网络中数据包乱序传输,接收方也能按照正确顺序重组数据。这对于应用程序处理数据的顺序性要求至关重要。

4. 应用场景

由于 TCP 提供可靠的数据传输和流控制,适用于以下应用场景:

  • 网页浏览 (HTTP/HTTPS):浏览器与服务器之间的数据传输需要高可靠性,以确保网页内容的完整性。
  • 文件传输 (FTP):文件传输协议需要保证数据的完整性和顺序性,避免文件损坏。
  • 电子邮件 (SMTP/IMAP/POP3):邮件传输需要确保邮件内容的完整性和可靠性。
  • 远程登录 (SSH/Telnet):远程登录需要可靠的数据传输,以确保命令和响应的正确性。
二、UDP 的特点及应用场景

用户数据报协议 (UDP, User Datagram Protocol) 是一种无连接的、不可靠的传输协议,适用于对实时性要求高但容错性强的应用场景。以下是 UDP 的主要特点和应用场景:

1. 无连接

UDP 是无连接协议,数据包直接发送到目标地址,无需建立连接。这使得 UDP 的传输过程更简单、更快速。

2. 不可靠性

UDP 不保证数据包的可靠传输,没有序列号和确认应答机制,数据包可能丢失、重复或乱序到达。应用程序需要自行处理数据包的可靠性。

3. 轻量级

UDP 头部开销小(仅 8 字节),数据包传输更加高效,适用于对实时性要求高的应用。

4. 支持广播和多播

UDP 支持广播和多播,允许数据包同时发送到多个接收方,适用于某些特殊应用场景。

5. 应用场景

由于 UDP 的不可靠性和高效性,适用于以下应用场景:

  • 实时音视频传输 (VoIP、视频会议):实时应用对时延敏感,允许少量数据丢失,但要求数据快速传输。
  • 在线游戏:在线游戏需要快速响应,允许少量数据丢失,以确保游戏体验。
  • 简单的请求-响应服务 (DNS):DNS 查询对时延要求高,允许少量查询重试。
  • 广播和多播:如 IPTV、实时股票行情等应用需要将数据包发送到多个接收方。
三、TCP 和 UDP 的区别

以下是 TCP 和 UDP 在不同方面的详细区别:

特性TCPUDP
连接类型面向连接无连接
可靠性可靠不可靠
数据传输顺序保证按序传输不保证按序传输
流控制具有流控制机制无流控制机制
拥塞控制具有拥塞控制机制无拥塞控制机制
头部开销较大(20-60 字节)较小(8 字节)
传输效率较低(由于可靠性机制)较高
适用场景需要高可靠性的数据传输需要高实时性的数据传输
应用示例HTTP、FTP、SMTP、SSHVoIP、视频会议、DNS、在线游戏
四、TCP 和 UDP 的工作原理
1. TCP 的工作原理
三次握手

三次握手用于建立 TCP 连接,确保双方都准备好进行数据传输:

  1. SYN:客户端发送 SYN 报文,表示请求建立连接。
  2. SYN-ACK:服务器收到 SYN 报文后,回应 SYN-ACK 报文,表示同意建立连接。
  3. ACK:客户端收到 SYN-ACK 报文后,发送 ACK 报文,表示确认建立连接。
数据传输

TCP 通过序列号和确认应答机制,确保数据包按序到达和完整性:

  • 序列号:每个数据包都有一个序列号,表示数据包在整个数据流中的位置。
  • 确认应答 (ACK):接收方发送 ACK 报文,确认收到的数据包。
四次挥手

四次挥手用于关闭 TCP 连接,确保双方都同意结束数据传输:

  1. FIN:客户端发送 FIN 报文,表示请求关闭连接。
  2. ACK:服务器收到 FIN 报文后,回应 ACK 报文,表示确认关闭请求。
  3. FIN:服务器发送 FIN 报文,表示同意关闭连接。
  4. ACK:客户端收到 FIN 报文后,发送 ACK 报文,表示确认关闭。
2. UDP 的工作原理
数据传输

UDP 直接将数据包发送到目标地址,无需建立连接和确认应答:

  • 数据报:每个数据包称为数据报,包含源地址、目标地址、数据等信息。
  • 无序传输:数据报可能乱序到达,应用程序需要自行处理数据的顺序性。
广播和多播

UDP 支持广播和多播,允许数据包同时发送到多个接收方:

  • 广播:将数据包发送到同一网络中的所有主机。
  • 多播:将数据包发送到特定的一组主机。
五、实例比较

以下是 TCP 和 UDP 的简单应用示例,展示它们的实际使用情况。

TCP 示例
TCP 服务器
#include <iostream>
#include <cstring>
#include <unistd.h>
#include <arpa/inet.h>

#define PORT 8080
#define BUFFER_SIZE 1024

int main() {
    int server_fd, new_socket;
    struct sockaddr_in address;
    int opt = 1;
    int addrlen = sizeof(address);
    char buffer[BUFFER_SIZE] = {0};
    const char* message = "Hello from server";

    // 创建套接字
    if ((server_fd = socket(AF_INET, SOCK_STREAM, 0)) == 0) {
        perror("socket failed");
        exit(EXIT_FAILURE);
    }

    // 绑定套接字
    if (setsockopt(server_fd, SOL_SOCKET, SO_REUSEADDR | SO_REUSEPORT, &opt, sizeof(opt))) {
        perror("setsockopt");
        exit(EXIT_FAILURE);
    }
    address.sin_family = AF_INET;
    address.sin_addr.s_addr = INADDR_ANY;
    address.sin_port = htons(PORT);

    if (bind(server_fd, (struct sockaddr*)&address, sizeof(address)) < 0) {
        perror("bind failed");
        exit(EXIT_FAILURE);
    }

    // 监听连接
    if (listen(server_fd, 3) < 0) {
        perror("listen");
        exit(EXIT_FAILURE);
    }

    // 接受连接
    if ((new_socket = accept(server_fd, (struct sockaddr*)&address, (socklen_t*)&addrlen)) < 0) {
        perror("accept");
        exit(EXIT_FAILURE);
    }

    read(new_socket, buffer, BUFFER_SIZE);
    std::cout << "Message received: " << buffer << std::endl;

    send(new_socket, message, strlen(message), 0);
    std::cout << "Hello message sent" << std::endl;

    close(new_socket);
    close(server_fd);
    return 0;
}
TCP 客户端
#include <iostream>
#include <cstring>
#include <unistd.h>
#include <arpa/inet.h>

#define PORT 8080
#define BUFFER_SIZE 1024

int main() {
    int sock = 0;
    struct sockaddr_in serv_addr;
    char buffer[BUFFER_SIZE] = {0};
    const char* message = "Hello from client";

    // 创建套接字
    if ((sock = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
        std::cerr << "Socket creation error" << std::endl;
        return -1;
    }

    serv_addr.sin_family = AF_INET;
    serv_addr.sin_port = htons(PORT);

    // 将服务器地址转为二进制形式
    if (inet_pton(AF_INET, "127.0.0.1", &serv_addr.sin_addr) <= 0) {
        std::cerr << "Invalid address/ Address not supported" << std::endl;
        return -1;
    }

    // 连接到服务器
    if (connect(sock, (struct sockaddr*)&serv_addr, sizeof(serv_addr)) < 0) {
        std::cerr << "Connection Failed" << std::endl;
        return -1;
    }

    send(sock, message, strlen(message), 0);
    std::cout << "Hello message sent" << std::endl;

    read(sock, buffer, BUFFER_SIZE);
    std::cout << "Message received: " << buffer << std::endl;

    close(sock);
    return 0;
}

运行结果

// 服务器端输出
Message received: Hello from client
Hello message sent

// 客户端输出
Hello message sent
Message received: Hello from server
UDP 示例
UDP 服务器
#include <iostream>
#include <cstring>
#include <unistd.h>
#include <arpa/inet.h>

#define PORT 8080
#define BUFFER_SIZE 1024

int main() {
    int sockfd;
    struct sockaddr_in server_addr, client_addr;
    char buffer[BUFFER_SIZE];
    const char* message = "Hello from server";

    // 创建套接字
    if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
        perror("socket creation failed");
        exit(EXIT_FAILURE);
    }

    memset(&server_addr, 0, sizeof(server_addr));
    memset(&client_addr, 0, sizeof(client_addr));

    // 填充服务器信息
    server_addr.sin_family = AF_INET;
    server_addr.sin_addr.s_addr = INADDR_ANY;
    server_addr.sin_port = htons(PORT);

    // 绑定套接字
    if (bind(sockfd, (const struct sockaddr *)&server_addr, sizeof(server_addr)) < 0) {
        perror("bind failed");
        close(sockfd);
        exit(EXIT_FAILURE);
    }

    socklen_t len = sizeof(client_addr);
    int n = recvfrom(sockfd, (char *)buffer, BUFFER_SIZE, MSG_WAITALL, (struct sockaddr *)&client_addr, &len);
    buffer[n] = '\0';
    std::cout << "Message received: " << buffer << std::endl;

    sendto(sockfd, message, strlen(message), MSG_CONFIRM, (const struct sockaddr *) &client_addr, len);
    std::cout << "Hello message sent" << std::endl;

    close(sockfd);
    return 0;
}
UDP 客户端
#include <iostream>
#include <cstring>
#include <unistd.h>
#include <arpa/inet.h>

#define PORT 8080
#define BUFFER_SIZE 1024

int main() {
    int sockfd;
    struct sockaddr_in serv_addr;
    char buffer[BUFFER_SIZE];
    const char* message = "Hello from client";

    // 创建套接字
    if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
        perror("socket creation failed");
        exit(EXIT_FAILURE);
    }

    memset(&serv_addr, 0, sizeof(serv_addr));

    // 填充服务器信息
    serv_addr.sin_family = AF_INET;
    serv_addr.sin_port = htons(PORT);
    serv_addr.sin_addr.s_addr = INADDR_ANY;

    sendto(sockfd, message, strlen(message), MSG_CONFIRM, (const struct sockaddr *) &serv_addr, sizeof(serv_addr));
    std::cout << "Hello message sent" << std::endl;

    socklen_t len = sizeof(serv_addr);
    int n = recvfrom(sockfd, (char *)buffer, BUFFER_SIZE, MSG_WAITALL, (struct sockaddr *) &serv_addr, &len);
    buffer[n] = '\0';
    std::cout << "Message received: " << buffer << std::endl;

    close(sockfd);
    return 0;
}

运行结果

// 服务器端输出
Message received: Hello from client
Hello message sent

// 客户端输出
Hello message sent
Message received: Hello from server
六、总结

通过对 TCP 和 UDP 的深入解析,我们可以得出以下结论:

  • TCP 提供可靠的数据传输,适用于需要高可靠性和顺序保证的应用,如网页浏览、文件传输、电子邮件和远程登录。
  • UDP 提供不可靠但高效的数据传输,适用于对实时性要求高但容错性强的应用,如实时音视频传输、在线游戏、DNS 查询和广播多播应用。

        在实际项目中,选择合适的传输协议至关重要,需要根据具体应用场景的需求来做出决策。希望本文的深入解析能够帮助读者更好地理解 TCP 和 UDP 的区别及其应用场景,提升网络编程能力。

标签:UDP,addr,C++,server,学懂,TCP,include,数据包
From: https://blog.csdn.net/martian665/article/details/141355413

相关文章

  • 学懂C++(四十):网络编程——深入详解 HTTP、HTTPS 及基于 Windows 系统的 C++ 实现
    目录一、引言二、HTTP协议1.HTTP概述2.HTTP工作原理3.HTTP请求和响应格式HTTP请求格式4.HTTP状态码三、HTTPS协议1.HTTPS概述2.HTTPS工作原理四、基于Windows系统的C++实现1.准备工作2.HTTP客户端实现示例代码3.HTTPS客户端实现示例代......
  • 2024年全国青少年信息素养大赛国赛PYTHON组(C++做法)
    目录前提第一题第二题第三题第四题第五题:第六题前提鄙人是C++学生,所以将PYTHON题做为C++题,还请各位多多海涵!!!部分芝士来自度娘和其它网站温馨提示:题目顺序可能不同,请各位仔细浏览! 第一题题目描述蓝蓝最近学到了一些单词,比如orange(橘子),apple(苹果),pear(梨)。......
  • 【C++篇】迈入新世界的大门——初识C++(下篇)
    文章目录前言引用引用的概念和定义引用的特性引用的使用const引用指针和引用的关系inline#define定义宏inlinenullptr前言接上篇:【C++篇】迈入新世界的大门——初识C++(上篇)引用引用的概念和定义引⽤不是新定义⼀个变量,⽽是给已存在变量取了⼀个别名,编译器不会......
  • C++类型推导
    C++11引入了auto和decltype关键字实现类型推导,通过这两个关键字不仅能方便地获取复杂的类型,还能简化书写,提高编码效率。1.auto类型推导1.1auto关键字的新意义在go语言中,在方法范围中声明的变量可以具有类型推导,例如:vari=10;//在go中自动类型推导变量i为int型而C++11......
  • C++第十一弹 -- STL之List的剖析与使用
    文章索引前言1.list的介绍2list的使用2.1list的构造函数2.2iterator的使用2.3listcapacity2.4listelementaccess2.5listmodifiers3.list的迭代器失效4.list与vector的对比总结前言本篇我们旨在探讨对于STL中list的使用,下一篇我们将会对list进行底层......
  • C++面试基础系列-volatile
    系列文章目录文章目录系列文章目录C++面试基础系列-volatile1.volatile核心规则2.C与C++中volatile区别2.1.C语言中的volatile2.2.C++中的volatile2.3.原子性和顺序2.4.易失性2.5.优化2.6.使用场景2.7.C++特有的特性2.8.C++20引入的变化(如果有)3.volatile常见面试问题4......
  • ArchLinux配置OpenCV C++环境
    本文将简单介绍在ArchLinux中安装OpenCVC++库并运行一个简单的OpenCV程序的过程。参考:https://github.com/donaldssh/Install-OpenCV我的环境最新的ArchLinuxKDEPlasma6桌面环境OpenCV4.10.0clang18.1.8gcc14.2.1安装安装以下包:sudopacman-Shdf5vtk......
  • 【C++】类与对象篇一
    【C++】类与对象篇一一.面向过程和面向对象初步认识二.类的详解1.类的引入2.类的定义3.类的访问限定符及封装(面试题)4.类的作用域5.类的实例化6.类对象模型三.结构体内存对齐规则(面试题)四.this指针1.this指针的特性2.this指针的(面试题)一.面向过程和面向对象......
  • 3142:练23.4 首字母(C、C++、python)
    3142:练23.4 首字母信息学奥赛一本通-编程启蒙(C++版)在线评测系统C源码:#include<stdio.h>#include<stdlib.h>intmain(){ charb; scanf("%c",&b); if(b=='a'){ printf("apple"); } elseif(b=='b'){ printf("ba......
  • C++运算符重载
    文章目录一、运算符重载1、规定2、operator关键词的使用二、赋值运算符的重载1、功能2、使用一、运算符重载1、规定C++允许我们对类类型使用运算符,但要我们自己通过运算符重载完成类类型的运算,如果没有对应的运算符重载就会报错。运算符重载需要使用特殊关键词......