进程之间的数据交互(使用管道和套接字实现进程通信)

news2024/11/24 1:38:20

本篇目录

  • 进程通信的方法有哪些
  • C语言使用管道实现进程通信
    • 使用管道进行实时通信
  • C语言使用套接字进行进程通信

进程通信的方法有哪些

进程通信是指在操作系统中,不同进程之间进行数据传递、信息共享和协调工作的方法。以下是常见的进程通信方法:

  1. 管道(Pipe):

    • 管道是一种基于文件描述符的通信方式,用于父子进程或具有共同祖先的进程之间的通信。
    • 管道可以使用无名管道(Unnamed Pipe)或命名管道(Named Pipe)实现。
  2. 消息队列(Message Queues):

    • 消息队列是一种通过消息传递进行通信的机制,其中消息具有特定的格式和标识符。
    • 进程可以通过将消息放入队列中来发送消息,其他进程则可以从队列中接收消息。
  3. 共享内存(Shared Memory):

    • 共享内存是一种进程间共享内存区域的通信方式,多个进程可以访问和修改相同的共享内存段。
    • 进程可以通过读写共享内存区域来实现数据的共享和传递。
  4. 信号量(Semaphores):

    • 信号量是一种用于进程间互斥和同步的通信方式,可以用于保证临界资源的互斥访问和进程的同步执行。
  5. 套接字(Socket):

    • 套接字是一种网络编程中常用的进程间通信方式,可用于不同主机之间的进程通信。
    • 套接字提供了一种可靠的、面向连接的通信机制,可以在客户端和服务器之间进行数据传输。
  6. 文件(File):

    • 进程可以使用文件作为一种简单的通信方法,通过读写文件来实现进程间的数据交换。
    • 这种通信方式适用于多个进程需要共享访问的持久化数据。

这些是常见的进程通信方法,每种方法都有其特点和适用场景。根据具体的需求和情况,选择合适的进程通信方法可以有效地实现进程间的数据传递和协作。

本文主要介绍 管道进程通信和套接字进程通信。

C语言使用管道实现进程通信

当在Linux命令行下输入ls并按下回车时,操作系统会执行以下步骤:

  1. 首先,命令解析器(通常是bash shell)会接收到输入的ls命令。
  2. 命令解析器会解析命令,并确定要执行的程序文件的路径。对于ls命令来说,该路径通常是/bin/ls/usr/bin/ls
  3. 接下来,操作系统会加载并执行ls程序。
  4. ls程序开始执行后,它会与操作系统进行交互。它访问当前目录的文件系统,并获取该目录下的所有文件和子目录的信息。
  5. ls程序将获取到的文件和目录信息输出到命令行窗口上,供用户查看。

总结来说,当在Linux命令行下输入ls并按下回车时,操作系统会加载并执行ls程序,然后ls程序会获取当前目录的文件和目录信息,并将其输出到命令行窗口上供用户查看。在 Windows 系统上,也是类似的道理。

命令的本质就是一个可执行的程序,在命令行下执行命令时,操作系统就会为该程序创建一个进程。当我们在命令行下输入命令,操作系统为我们执行命令,此时操作系统会在当前目录下寻找该命令的可执行文件,如果没有找到,则去环境变量下寻找。当程序(命令)被执行时,程序的输出都会显示在命令行下。

以下是一个进程通信的C语言例子:在下面这个 C 语言代码中,popen("command", "r"); 函数会调用另外一个程序,这里的 command 就是一个可执行程序的名字,当该程序被调用时,操作系统会为它创建进程。而下面的代码编译后运行时,也是一个进程,该进程在运行时,会创建对应 command 的进程,随后 command 的进程的输出信息会通过管道被它的调用进程读取,并将信息输出。

#include <stdio.h>

int main() {
    FILE *fp;
    char buffer[256];

    // 执行命令并打开命令输出管道
    fp = popen("your_command", "r");

    if (fp == NULL) {
        printf("无法执行命令\n");
        return 1;
    }

    // 逐行读取命令输出信息
    while (fgets(buffer, sizeof(buffer), fp) != NULL) {
        // 处理每行输出
        // 示例:打印输出
        printf("%s", buffer);
    }

    // 关闭命令输出管道
    pclose(fp);

    return 0;
}

在上面的示例中,关键步骤如下:

  1. 使用popen函数执行指定的命令,并打开命令输出管道。需要将your_command替换为实际的命令。
  2. 使用fgets函数从命令输出管道中逐行读取输出信息到缓冲区buffer中。
  3. 在每行输出信息的处理过程中,可以根据需要进行相应的操作,例如打印输出或做其他处理。
  4. 使用pclose函数关闭命令输出管道。

例如将 command 替换成 ls (Linux下) 或者 dir (Windows下) 运行:

在这里插入图片描述
这样就实现了进程通信,ls 或者 dir 的输出信息传输给调用它们的进程。

在上面的代码中,调用了 ls,调用进程和被调用进程是父子进程的关系。管道通信只能在具有父子关系或共同祖先的进程之间进行。如果你需要在没有父子关系的两个独立进程之间进行实时通信,可能需要使用其他IPC机制,比如命名管道、消息队列、共享内存等。

当我们使用集成的开发环境(IDE Integrated Development Environment) IntelliJ IDEA 去做Java 开发的时候,IntelliJ IDEA 做的就是调用 javac 去编译 java 的源代码,当源代码有错误的时候,IntelliJ IDEA 会把错误信息显示在图形界面上,IntelliJ IDEA 的源代码中就用到了进程通信。它调用了 javac,此时 javac 就是它的子进程,然后 javac 把错误信息通过管道传给它的调用进程。

在这里插入图片描述

使用管道进行实时通信

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/wait.h>

int main() {
    int pipefd[2];
    pid_t pid;
    
    // 创建管道
    if (pipe(pipefd) == -1) {
        perror("pipe");
        exit(EXIT_FAILURE);
    }
    
    // 创建子进程
    pid = fork();
    
    if (pid > 0) {
        // 父进程
        
        // 关闭写端
        close(pipefd[1]);
        
        // 从读端读取数据
        char buffer[1024];
        ssize_t bytesRead = read(pipefd[0], buffer, sizeof(buffer));
        printf("父进程接收到消息:%.*s\n", (int)bytesRead, buffer);
        
        // 等待子进程结束
        wait(NULL);
        
        // 关闭读端
        close(pipefd[0]);
    }
    else if (pid == 0) {
        // 子进程
        
        // 关闭读端
        close(pipefd[0]);
        
        // 向写端写入数据
        char message[] = "Hello, parent process!";
        write(pipefd[1], message, sizeof(message));
        
        // 关闭写端
        close(pipefd[1]);
    }
    else {
        // fork失败
        perror("fork");
        exit(EXIT_FAILURE);
    }
    
    return 0;
}

这段代码是一个使用管道进行父子进程间通信的示例。

首先,在 main 函数中创建了一个整型数组 pipefd 来存储管道的文件描述符,这个数组有两个元素,分别表示管道的读端和写端。

接下来调用 pipe() 函数创建管道。如果函数返回值为 -1,则说明创建管道失败,程序将打印错误信息并退出。

然后通过 fork() 函数创建子进程。如果返回值大于 0,说明是在父进程中;如果返回值等于 0,说明是在子进程中。

对于父进程,它关闭了管道的写端 pipefd[1],然后通过 read() 函数从管道的读端 pipefd[0] 中读取数据,将读取到的数据存储在缓冲区 buffer 中,并打印输出。

接着调用 wait(NULL) 函数等待子进程结束。

最后,父进程关闭管道的读端 pipefd[0],完成操作。

对于子进程,它关闭了管道的读端 pipefd[0],然后通过 write() 函数向管道的写端 pipefd[1] 中写入数据,该数据为字符数组 message 中的内容。

最后,子进程关闭管道的写端 pipefd[1],完成操作。

需要注意的是,代码中使用了一些系统调用函数如 close()read()write()wait(),需要包含相应的头文件 <unistd.h><sys/wait.h>。此外,还包含了 <stdio.h><stdlib.h> 头文件用于提供标准输入输出和一些基本函数。

pid = fork(); 这行代码的作用是通过调用 fork() 函数创建一个子进程。

在调用 fork() 函数时,会复制当前进程(称为父进程),并创建一个新的子进程。这个新的子进程与父进程几乎完全相同,包括代码、数据和打开的文件等。

具体而言,fork() 函数的返回值有以下三种可能情况:

  1. 如果返回值是负数,表示创建子进程失败。
  2. 如果返回值是 0,表示当前代码正在运行的是子进程。
  3. 如果返回值大于 0,表示当前代码正在运行的是父进程,返回值是新创建的子进程的进程ID。

根据返回值的不同,代码中的 if 语句可以分别对父进程和子进程做出不同的处理,实现父子进程之间的分支逻辑。

通常情况下,在 fork() 后的代码中会使用条件判断来区分父进程和子进程的执行路径,从而实现不同的操作或任务。

C语言使用套接字进行进程通信

代码一:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>

#define PORT 8888
#define MAX_BUFFER_SIZE 1024

int main() {
    int sockfd, newsockfd;
    struct sockaddr_in serverAddr, clientAddr;
    socklen_t addrLen;
    char buffer[MAX_BUFFER_SIZE];

    // 创建套接字
    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0) {
        perror("socket");
        exit(1);
    }

    // 设置服务器地址信息
    serverAddr.sin_family = AF_INET;
    serverAddr.sin_port = htons(PORT);
    serverAddr.sin_addr.s_addr = INADDR_ANY;

    // 绑定套接字到指定端口
    if (bind(sockfd, (struct sockaddr*)&serverAddr, sizeof(serverAddr)) < 0) {
        perror("bind");
        exit(1);
    }

    // 监听连接请求
    if (listen(sockfd, 10) < 0) {
        perror("listen");
        exit(1);
    }

    printf("Waiting for incoming connections...\n");

    // 接受连接请求
    addrLen = sizeof(clientAddr);
    newsockfd = accept(sockfd, (struct sockaddr*)&clientAddr, &addrLen);
    if (newsockfd < 0) {
        perror("accept");
        exit(1);
    }

    // 接收消息
    if (recv(newsockfd, buffer, MAX_BUFFER_SIZE, 0) < 0) {
        perror("recv");
        exit(1);
    }

    printf("Message received from client: %s\n", buffer);

    close(newsockfd);
    close(sockfd);

    return 0;
}

代码二:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>

#define PORT 8888

int main() {
    int sockfd;
    struct sockaddr_in serverAddr;
    char buffer[1024];

    // 创建套接字
    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0) {
        perror("socket");
        exit(1);
    }

    // 设置服务器地址信息
    serverAddr.sin_family = AF_INET;
    serverAddr.sin_port = htons(PORT);
    serverAddr.sin_addr.s_addr = INADDR_ANY;

    // 连接到服务器
    if (connect(sockfd, (struct sockaddr*)&serverAddr, sizeof(serverAddr)) < 0) {
        perror("connect");
        exit(1);
    }

    printf("Enter a message: ");
    fgets(buffer, 1024, stdin);

    // 发送消息到服务器
    if (send(sockfd, buffer, strlen(buffer), 0) < 0) {
        perror("send");
        exit(1);
    }

    printf("Message sent to the server: %s\n", buffer);

    close(sockfd);

    return 0;
}

编译运行:先编译运行代码一,再编译运行代码二。

在这里插入图片描述使用套接字,不仅可以实现本机的进程通信,也可以实现不同主机之间的进程通信。涉及计算机网络相关知识。

代码一解读:
这段代码是进程A的代码,它创建一个套接字并监听指定端口,等待来自进程B的连接请求。一旦连接建立,进程A将接收来自进程B发送的消息,并在控制台输出。当运行进程A时,它相当于一个服务器。

代码解读如下:

  1. 头文件包含部分:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>

这些头文件提供了套接字编程所需的函数、结构和常量。

  1. 定义常量:
#define PORT 8888                      // 指定监听端口号
#define MAX_BUFFER_SIZE 1024           // 指定缓冲区最大大小
  1. 主函数部分:
int main() {
    int sockfd, newsockfd;
    struct sockaddr_in serverAddr, clientAddr;
    socklen_t addrLen;
    char buffer[MAX_BUFFER_SIZE];

    // 创建套接字
    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0) {
        perror("socket");
        exit(1);
    }

通过 socket 函数创建一个套接字,AF_INET 表示使用 IPv4 地址族,SOCK_STREAM 表示使用 TCP 传输协议。如果创建失败,会打印错误信息并退出程序。

    // 设置服务器地址信息
    serverAddr.sin_family = AF_INET;
    serverAddr.sin_port = htons(PORT);
    serverAddr.sin_addr.s_addr = INADDR_ANY;

设置服务器地址信息,包括地址族、端口号和 IP 地址。INADDR_ANY 表示绑定到本地的所有可用网络接口。

    // 绑定套接字到指定端口
    if (bind(sockfd, (struct sockaddr*)&serverAddr, sizeof(serverAddr)) < 0) {
        perror("bind");
        exit(1);
    }

将套接字与指定的端口号进行绑定,如果绑定失败则打印错误信息并退出。

    // 监听连接请求
    if (listen(sockfd, 10) < 0) {
        perror("listen");
        exit(1);
    }

开始监听连接请求,指定最大允许排队等待连接的请求数量为 10。如果监听失败则打印错误信息并退出。

    printf("Waiting for incoming connections...\n");

    // 接受连接请求
    addrLen = sizeof(clientAddr);
    newsockfd = accept(sockfd, (struct sockaddr*)&clientAddr, &addrLen);
    if (newsockfd < 0) {
        perror("accept");
        exit(1);
    }

输出提示信息,并调用 accept 函数接受来自客户端的连接请求。如果接受失败则打印错误信息并退出。

    // 接收消息
    if (recv(newsockfd, buffer, MAX_BUFFER_SIZE, 0) < 0) {
        perror("recv");
        exit(1);
    }

使用 recv 函数从客户端接收消息,将消息存储在缓冲区 buffer 中。如果接收失败则打印错误信息并退出。

    printf("Message received from client: %s\n", buffer);

    close(newsockfd);
    close(sockfd);

    return 0;
}

将接收到的消息输出到控制台,并关闭套接字和新的套接字描述符,释放资源。最后返回 0 表示程序正常结束。

代码二解读:
代码二是一个简单的客户端程序,用于通过TCP/IP协议向服务器发送消息。

  1. 首先,代码包含了一些系统库头文件,如stdio.h、stdlib.h等,这些头文件提供了所需的函数和类型定义。
  2. 定义了常量PORT,指定服务器的端口号为8888。
  3. 主函数开始后,创建一个套接字sockfd,使用socket函数,该套接字用于与服务器进行通信。如果创建失败,则打印错误信息并退出程序。
  4. 设置服务器地址信息,结构体变量serverAddr是存储服务器地址的结构体,设置其中的成员变量sin_family表示地址家族为AF_INET(IPv4),sin_port表示端口号,使用htons函数将主机字节序转换为网络字节序,sin_addr.s_addr表示服务器的IP地址,使用宏INADDR_ANY表示可以使用任意可用的IP地址。
  5. 使用connect函数将套接字连接到指定的服务器地址,如果连接失败,则打印错误信息并退出程序。
  6. 打印提示信息,要求用户输入消息内容。
  7. 使用fgets函数从标准输入中读取用户输入的消息,保存在buffer数组中。
  8. 使用send函数将消息发送给服务器,参数分别为套接字描述符 sockfd、消息内容 buffer、消息长度 strlen(buffer),如果发送失败,则打印错误信息并退出程序。
  9. 打印已发送的消息内容。
  10. 关闭套接字,使用close函数关闭套接字描述符。
  11. 程序执行完毕,返回0作为退出状态码。

总体上,这段代码实现了与服务器建立连接,并向服务器发送消息的功能。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/745258.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux网络第五章——YUM和NFS共享服务—YUM

YUM简介&#xff1a; 是一个专门为了解决包的依赖关系而存在的软件包管理器。 可以一键下载&#xff0c;一键安装和卸载。yum 是改进型的 RPM 软件管理器&#xff0c; 它解决了 RPM 所面临的软件包依赖问题。 yum 在服务器端存有所有的 RPM 包&#xff0c;并将各个包之间的依…

相比传统ERP,基于项目的ERP系统有何不同?

服务型企业是独特的以人为本的组织&#xff0c;依靠员工来销售、配置和提供一系列的产品和服务&#xff0c;这通常是以项目或合同为基础。 在这样一个环境中&#xff0c;服务型企业的基本财务要求与传统的制造和供应链型企业有很大的不同。 基于项目的企业需要管理资源&#x…

2023最新 Navicat 16.2.3 安装和学习试用连接Redis教程详解

&#x1f337;&#x1f341; 博主 libin9iOak带您 Go to New World.✨&#x1f341; &#x1f984; 个人主页——libin9iOak的博客&#x1f390; &#x1f433; 《面试题大全》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33…

IIS8 安装 .net1.1

直接上正文&#xff1b; 一 安装补丁 安装顺序&#xff1a; 1 dotnetfx.exe 2 NDP1.1sp1-KB867460-X86 3 NDP1.1sp1-KB886903-X86 .NET_Framework_Cleanup_Tool.zip&#xff08;这个是卸载.net framework的软件&#xff0c;后面可能会用&#xff09; 链接&#xff1a;ht…

ubuntu软件商店换阿里源,并解决更新源报错-->无法验证下列签名: NO_PUBKEY 3B4FE6ACC0B21F32

目录 一、背景 二、给源文件备份 三、更新源 四、解决报错 五、继续更新源 六、完成更新&#xff0c;下载应用 一、背景 重装了个ubuntu&#xff0c;发现软件商店用不了&#xff0c;打算换源。 二、给源文件备份 1&#xff09;登录ubuntu系统&#xff0c;打开终端&…

mybatis双重foreach实现遍历map中的两个list数组

文章目录 实现背景&#xff1a;前端传值的格式Debug断点调试java如何解析json对象第一步 JSONArray.fromObject()第二步 遍历jsonArray第三步 mybatis双重foreach foreach标签说明最终效果 实现背景&#xff1a; 前端传值时可能会有多个字段传递过来&#xff0c;需要后台将这多…

SpringBoot + WebSocket+STOMP指定推送消息

目录 一、前提条件1.2 环境要求1.3 依赖 二、相关工具类准备2.1 发送消息载体2.2 接收消息载体2.3 消息处理接口2.4 为 STOMP 消息传递配置 Spring 三、前端部分四、效果 一、前提条件 本文将简单的描述SpringBoot WebSocketSTOMP指定推送消息场景&#xff0c;不包含信息安全加…

搭建FRP内网穿透服务器来远程访问本地windows/linux中的web服务

什么是FRP&#xff1f; FRP 是一个免费开源的用于内网穿透的反向代理应用&#xff0c;它支持 TCP、UDP 协议&#xff0c; 也为 http 和 https 协议提供了额外的支持。 FRP有服务端和客户端&#xff0c;你将服务端安装在你买的便宜云服务器上&#xff0c;FRP客户端安装在你自己的…

数据标注是什么?

关于数据标注您需要了解的一切——专家解答&#xff1a;澳鹏产品管理总监Meeta Dash 人工智能&#xff08;AI&#xff09;的质量取决于对其予以训练所使用的数据。由于训练数据的质量和数量直接决定AI算法的成败&#xff0c;因此&#xff0c;对于一个AI项目&#xff0c;平均80…

简单线性回归评估指标+R Squared

使得每一个数据集尽可能的小 均方误差MSE&#xff1a;&#xff08;平方和取平均值&#xff09; 均方根误差RMSE&#xff1a;&#xff08;平方和取平均值开根号&#xff09;&#xff1a;平均误差值 平均绝对误差MAE&#xff1a;&#xff08;绝对值取平均&#xff09;&#xff1a…

Misc题目总结

目录 [WUSTCTF2020]girlfriend 手机键盘隐写&#xff1a; [GUET-CTF2019]zips setup.sh文件的运行方法&#xff1a; [MRCTF2020]千层套路 重复解zip加密压缩包脚本&#xff1a; 根据RGB坐标构造二维码脚本&#xff1a; [DDCTF2018](╯□&#xff09;╯︵ ┻━┻ [WUSTC…

COT、COT-SC、TOT 大预言模型思考方式||底层逻辑:prompt设定

先讲一下具体缩写的意思 COT-chain of thoughts COT-SC (Self-consistency) Tree of thoughts:Deliberate problem solving with LLM 我理解其实不复杂 1. 最简单的是&#xff1a;直接大白话问一次 &#xff08;IO&#xff09; 2. 进阶一点是&#xff1a;思维链&#xff0c;…

wx.getUserProfile too frequently 问题

接口调用频率规范 概念介绍 小程序wx接口可分为“普通接口”和“限频接口”。 “限频接口”指的是一个用户在一段时间内不允许频繁调用的wx接口&#xff0c;此类接口一般会调用到微信后台系统资源&#xff0c;为了保护系统&#xff0c;同时防止用户资源被滥用&#xff0c;开…

C语言进阶之字符串函数和内存函数的介绍及部分函数的模拟实现

字符串函数和内存函数 1.字符串函数介绍1.1 strlen1.2 strcpy1.3 strcat1.4 strcmp1.5 strncpy1.6 strncat1.7 strncpy1.8 strstr1.9 strtok1.10 strerror1.11 字符分类函数 2.内存函数2.1 memcpy2.2 memmove2.3 memcmp 3.函数的模拟实现3.1 模拟实现strlen3.2 模拟实现strcpy3…

2023年上半年鲁大师手机流畅排行榜:OPPO Find X6 Pro稳居榜首,iQOO Neo8 Pro紧随其后

随着科技的不断进步&#xff0c;手机的流畅性在用户选择手机时也变得越来越重要。 在2023年上半年的手机流畅排行榜中&#xff0c;OPPO Find X6 Pro以219.98的得分稳居榜首。紧随其后的是iQOO Neo8 Pro&#xff0c;得分为219.73。 接下来&#xff0c;让我们一起来看看这份排行榜…

小白白也能学会的 PyQt 教程 —— 实战简易计算器

文章目录 导言一、从 0 到 1&#xff1a;简易计算器实现1、功能实现2、代码分析3、实现效果 二、由简入繁&#xff1a;优化简易计算器的 UI1、开始布局2、实现 Driver3、Driver 代码4、效果展示 总结 导言 PyQt5 是一个用于创建桌面应用程序的 Python 库&#xff0c;它提供了丰…

JavaWeb 拦截器(interceptor)和过滤器(filter)的区别 总结

一、区别 实现原理使用和适用范围使用场景触发时机请求顺序 二、具体展开说明 2.1 实现原理 过滤器Filter &#xff1a;基于函数也就是方法回调来完成拦截和放行的操作。 拦截器(interceptor)&#xff1a;拦截器则是通过Java 反射机制&#xff08;动态代理&#xff09;来实…

python_寻找底部股票

目录 写在前面&#xff1a; 下面开始进入正文&#xff0c;正文很长&#xff0c;先概述要点步骤&#xff0c;以防迷路。 先看最终结果&#xff08;当前价格为2023-07-10收盘价&#xff09; 1 下载股票基本信息、股票日数据 step one step two 2 从股票日数据中提取股票月数…

优思学院|如何衡量六西格玛中解决方案的有效性?

当你遇到任何质量问题&#xff0c;不论你使用的PDCA或者是六西格玛DMAIC策略&#xff0c;(1) 你都需要测量当前绩效&#xff0c;(2) 分析其原因&#xff0c;以至其根本原因&#xff0c;(3) 进行方案的试行、假设检验来进行验证&#xff0c;这就是解问题的必须步骤。 除了假设检…

计网笔记--运输层(vital)

1--运输层概述 运输层的任务&#xff1a; 为运行在不同主机上的应用进程提供直接的通信服务&#xff1b; 运输层为应用层提供了两种不同的运输协议&#xff1a; 面向连接的 TCP 和无连接的 UDP 协议&#xff1b; 2--端口号、复用与分用的概念 端口号&#xff1a; 端口号用于区分…