【Linux初阶】进程间通信介绍 管道

news2025/1/12 16:05:37

🌟hello,各位读者大大们你们好呀🌟
🍭🍭系列专栏:【Linux初阶】
✒️✒️本篇内容:进程间通信介绍,管道概述,匿名管道+应用,命名管道+应用
🚢🚢作者简介:计算机海洋的新进船长一枚,请多多指教( •̀֊•́ ) ̖́-


文章目录

  • 一、进程间通信介绍
    • 1.进程间通信目的
    • 2.进程间通信发展
    • 3.进程间通信分类
      • 管道
      • System V IPC
      • POSIX IPC
  • 二、管道概述
    • 1.什么是管道
    • 2.进程间通信的原理
    • 3.站在文件描述符角度-深度理解管道
  • 三、匿名管道
    • 1.匿名管道概述及代码实现
    • 2.管道的读写特征(4种情况)
    • 3.管道的特征(5种特征)
  • 四、匿名管道应用
    • 1.匿名管道在命令行中的应用
    • 2.基于匿名管道的进程池设计
  • 五、命名管道
    • 1.命名管道概述
    • 2.创建一个命名管道
      • (1)命令行创建命名管道
      • (2)程序创建命名管道
    • 3.匿名管道与命名管道的区别
    • 4.命名管道的打开规则
  • 六、用命名管道实现server&client通信
  • 结语


一、进程间通信介绍

1.进程间通信目的

  • 数据传输:一个进程需要将它的数据发送给另一个进程(进程具有独立性,通信成本不低)。
  • 资源共享:多个进程之间共享同样的资源。
  • 通知事件:一个进程需要向另一个或一组进程发送消息,通知它(它们)发生了某种事件(如进程终止时要通知父进程)。
  • 进程控制:有些进程希望完全控制另一个进程的执行(如Debug进程),此时控制进程希望能够拦截另一个进程的所有陷入和异常,并能够及时知道它的状态改变。

2.进程间通信发展

  • 管道
  • System V进程间通信(聚焦本地通信)
  • POSIX进程间通信(让通信过程可以跨主机)

3.进程间通信分类

管道

  • 匿名管道pipe
  • 命名管道

System V IPC

  • System V 消息队列
  • System V 共享内存
  • System V 信号量

POSIX IPC

  • 消息队列
  • 共享内存
  • 信号量
  • 互斥量
  • 条件变量
  • 读写锁

二、管道概述

1.什么是管道

  • 管道是Unix中最古老的进程间通信的形式。
  • 我们把从一个进程连接到另一个进程的一个数据流称为一个“管道"
  • 管道分匿名管道命名管道

2.进程间通信的原理

我们知道,进程间通信需要一定的成本,那么这个成本是怎么产生的呢?这就要我们理解通信的本质了:1.OS直接或间接给通信双方的进程提供“内存空间”2.要通信的进程,必须看到同一份公共的资源

这个很好理解,因为进程具有独立性,所以需要有一个公共的地方让两个进程完成信息或资源的传输。我们所学的不同的通信种类,就是以操作系统提供的不同模块进行区别分类的

任何一个文件(struct file)都有 file的操作方法、属于自己的内核缓冲区strcut Page[]。当我们进行 fork创建子进程的时候,我们会将父进程的 task_strcut文件描述符表拷贝一份,子进程文件描述符表指向的文件在不做修改的情况下和父进程文件描述符表指向的文件相同。此时,我们就能让父子两个不同的进程看到同一份资源了。
在这里插入图片描述

在能看到同一份资源的基础上,如果父进程可以向文件缓冲区中写入,子进程能从文件缓冲区中读取,不就完成了进程间的通信了吗。我们将操作系统提供的这个内核级文件,我们称之为管道文件

进程间通信的数据不会刷新到磁盘,而是在OS内部完成,因为刷新到磁盘会大大降低通信的速度。管道文件是一个内存级文件,通信只需要在OS内部完成,因此我们可以不关心它在磁盘的路径,甚至这个文件可以是虚拟出来的。

3.站在文件描述符角度-深度理解管道

下面我们一起来认识一下管道通信的基本原理:

在这里插入图片描述

上图的步骤执行完之后,就可以实现将父进程的数据传输到子进程中去了。通过对上面图解的学习,我们不难知道:管道通常只能进行单向数据通信!单向通信相比于双向通信更简单且更具有依赖性。

最终,设计者们根据这个方案的特性(有出入口、单向传输),为这个方案/方法取名为 - 管道


三、匿名管道

1.匿名管道概述及代码实现

我们知道通信的实现需要两个步骤:1.让不同的进程拥有一份共享资源;2.通信。在文章之前的内容中,我们已经对第一点的原理进行了讲解,下面我将带着大家重一起学习管道的具体实现(开辟共享资源+通信)。

匿名管道:目前能用来进行父子进程间的通信

#include <unistd.h>
功能:创建一无名管道
原型
int pipe(int fd[2]);
参数
fd:文件描述符数组,其中fd[0]表示读端, fd[1]表示写端
返回值:成功返回0,失败返回错误代码

因为文件描述符0,1,2已经被占用,因此在理想情况下也需要用3,4代表读取和写入端,由于我们不清楚系统申请读写的顺序,所以我们习惯上直接使用fd[0]表示读端, fd[1]表示写端。

读取和写入端巧记:[0]: 读取,嘴巴,读书的;[1]: 写入,钢笔,写的

代码示例:

#include <iostream>
#include <cstdio> //snprintf
#include <cstring> //write
#include <string>
#include <cassert>
#include <unistd.h> //sleep wait
#include <sys/types.h>
#include <sys/wait.h>


using namespace std;

// 父进程进行读取,子进程进行写入
int main()
{
    // 第一步:创建管道文件,打开读写端
    int fds[2];
    int n = pipe(fds);
    assert(n == 0);

    // 第二步: fork
    pid_t id = fork();
    assert(id >= 0);
    if (id == 0)
    {
        // 子进程进行写入 - 关闭读
        close(fds[0]);
        // 子进程的通信代码
        const char* s = "我是子进程,我正在给你发消息";
        int cnt = 0;
        while (true)
        {
            cnt++;
            char buffer[1024]; // 只有子进程能看到!
            snprintf(buffer, sizeof buffer, "child->parent say: %s[%d][%d]", s, cnt, getpid());//格式化输出

            // 写端写满的时候,再写会阻塞,等对方进行读取!
            write(fds[1], buffer, strlen(buffer));
            cout << "count: " << cnt << endl;

			//sleep(2); - 用于测试写端阻塞
        }

        // 子进程
        close(fds[1]); // 子进程关闭写端fd
        cout << "子进程关闭自己的写端" << endl;

        exit(0);
    }
    
    // 父进程进行读取
    close(fds[1]);
    // 父进程的通信代码
    while (true)
    {
        //sleep(2); - 用于测试读端阻塞
        char buffer[1024];

        // 如果管道中没有了数据,读端在读,默认会直接阻塞当前正在读取的进程!
        ssize_t s = read(fds[0], buffer, sizeof(buffer) - 1); //buffer如果满了,s代表下一个未被使用的字节数

        if (s > 0)
        {
            buffer[s] = 0;
            cout << "Get Message# " << buffer << " | my pid: " << getpid() << endl;
        }
        else if (s == 0)
        {
            //读到文件结尾 - 读完之后s为0 - 跳出循环结束读取
            cout << "read: " << s << endl;
            break;
        }
        break;

    }
    close(fds[0]);
    cout << "父进程关闭读端" << endl;

    int status = 0;
    n = waitpid(id, &status, 0);
    assert(n == id);

    cout << "pid->" << n << " : " << (status & 0x7F) << endl;

    return 0;
}

在这里插入图片描述
———— 我是一条知识分割线 ————

2.管道的读写特征(4种情况)

读写特征:

  1. 管道是一个固定大小的缓冲区

  2. 读慢、写快:写端将缓冲区写满的时候,再写(写端)会阻塞,需要等对方进行读取!

  3. 读快、写慢:如果管道中没有了数据,读端在读,默认会直接阻塞当前正在读取的进程!

  4. 写关闭、读到0:根据自己代码的判断,进行不同的操作;
    在这里插入图片描述

  5. 读关闭、一直写:为避免资源浪费,OS会关闭写端,即给写进程发送信号,终止写端。

———— 我是一条知识分割线 ————

3.管道的特征(5种特征)

  1. 管道的生命周期随进程;
  2. 管道可以用来实现具有血缘关系(父子、兄弟)进程之间的通信,常用于父子进程通信;
  3. 管道是面向字节流的(网路);
  4. 一般而言,内核会对管道操作进行同步与互斥,同步与互斥机制是一种对共享资源的一种保护方案;
  5. 管道是半双工的,即数据只能向一个方向流动;需要双方通信时,需要建立起两个管道;
    在这里插入图片描述

四、匿名管道应用

1.匿名管道在命令行中的应用

我们在命令行中经常会使用这样的指令

cat file.txt | grep "hello"

它实质上就是通过树划线对指令进行了分割,让操作系统 fork两个子进程对分割好的两条指令进行实现,再通过兄弟管道之间的进程通信,最终变成我们看到的样子的!

2.基于匿名管道的进程池设计

我们可以用匿名管道设计一个简易的进程池,即用一个辅助进程控制其他进程完成工作。在父进程(写端)我们通过给不同的子进程(读端)传输不同的命令操作码(commandCode),让对应的子进程根据命令操作码实现不同的功能,我们让命令操作码为 4字节。

场景示例:我们可以给 2号进程发送命令操作码和对应的运算数据,唤醒 2号进程之后,让进程对命令操作码对应的功能(运算)进行实现。

我们希望:我们可以将我们的任务均衡的下发给每一个子进程,让子进程执行 - 负载均衡(单机)。

在这里插入图片描述

代码示例

  • Makefile
processpool:processPool.cc
g++ - o $@ $ ^ -std = c++11
.PHONY:clean
clean :
rm - rf processpool
  • ProcessPool.cc
#include <iostream>
#include <string>
#include <vector>
#include <cstdlib>
#include <cassert>
#include <ctime>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>

#define MakeSeed() srand((unsigned long)time(nullptr) ^ getpid() ^ 0x171237 ^ rand() % 1234)//随机数种子

#define PROCSS_NUM 10

///子进程要完成的某种任务 -- 模拟一下/
// 函数指针 类型
typedef void (*func_t)();

void downLoadTask()
{
    std::cout << getpid() << ": 下载任务\n"
        << std::endl;
    sleep(1);
}

void ioTask()
{
    std::cout << getpid() << ": IO任务\n"
        << std::endl;
    sleep(1);
}

void flushTask()
{
    std::cout << getpid() << ": 刷新任务\n"
        << std::endl;
    sleep(1);
}

void loadTaskFunc(std::vector<func_t>* out) //加载方法表
{
    assert(out);
    out->push_back(downLoadTask);
    out->push_back(ioTask);
    out->push_back(flushTask);
}

/下面的代码是一个多进程程序//
class subEp // Endpoint
{
public:
    subEp(pid_t subId, int writeFd)
        : subId_(subId), writeFd_(writeFd)
    {
        char nameBuffer[1024];  //定义对象名称
        snprintf(nameBuffer, sizeof nameBuffer, "process-%d[pid(%d)-fd(%d)]", num++, subId_, writeFd_);
        name_ = nameBuffer;
    }

public:
    static int num; //用于区别不同对象(整体)
    std::string name_;  //管道+子进程的整体名称
    pid_t subId_;   //子进程pid
    int writeFd_;   //父进程写端的文件描述符
};

int subEp::num = 0;

int recvTask(int readFd) //读端接口
{
    int code = 0;
    ssize_t s = read(readFd, &code, sizeof code);
    if (s == 4) return code;
    else if (s <= 0) return -1;
    else return 0;
}

void sendTask(const subEp& process, int taskNum)
{
    std::cout << "send task num: " << taskNum << " send to -> " << process.name_ << std::endl;
    int n = write(process.writeFd_, &taskNum, sizeof(taskNum));
    assert(n == sizeof(int));
    (void)n;
}

void createSubProcess(std::vector<subEp>* subs, std::vector<func_t>& funcMap)
{
    std::vector<int> deleteFd;  //管道+子进程整体选择(可以选择不同整体)
    for (int i = 0; i < PROCSS_NUM; i++)
    {
        int fds[2];
        int n = pipe(fds);
        assert(n == 0);
        (void)n;
        // 父进程打开的文件,是会被子进程共享的
        // 你试着多想几轮 - 子进程会继承上一个子进程的写端
        pid_t id = fork();
        if (id == 0)
        {
            for (int i = 0; i < deleteFd.size(); i++) close(deleteFd[i]); //关闭从上一个子进程继承而来的写端
            // 子进程, 进行处理任务
            close(fds[1]);
            while (true)
            {
                // 1. 获取命令码,如果没有发送,我们子进程应该阻塞
                int commandCode = recvTask(fds[0]);
                // 2. 完成任务
                if (commandCode >= 0 && commandCode < funcMap.size())
                    funcMap[commandCode]();
                else if (commandCode == -1) break;
            }
            exit(0);
        }
        close(fds[0]);
        subEp sub(id, fds[1]);  //构造一个整体对象
        subs->push_back(sub);   //将新整体链接到vector中
        deleteFd.push_back(fds[1]); //保存文件描述符,方便以后关闭从上一个子进程继承而来的写端
    }
}

void loadBlanceContrl(const std::vector<subEp>& subs, const std::vector<func_t>& funcMap, int count)
{
    int processnum = subs.size();
    int tasknum = funcMap.size();
    bool forever = (count == 0 ? true : false);

    while (true)
    {
        // 1. 选择一个子进程 --> std::vector<subEp> -> index - 随机数(负载均衡)
        int subIdx = rand() % processnum;
        // 2. 选择一个任务 --> std::vector<func_t> -> index
        int taskIdx = rand() % tasknum;
        // 3. 任务发送给选择的进程
        sendTask(subs[subIdx], taskIdx);
        sleep(1);   //防止发送过快
        if (!forever)
        {
            count--;
            if (count == 0) break;
        }
    }
    // write quit -> read 0
    for (int i = 0; i < processnum; i++) close(subs[i].writeFd_); // waitpid();
}


void waitProcess(std::vector<subEp> processes)
{
    int processnum = processes.size();
    for (int i = 0; i < processnum; i++)
    {
        waitpid(processes[i].subId_, nullptr, 0);
        std::cout << "wait sub process success ...: " << processes[i].subId_ << std::endl;
    }
}

int main()
{
    MakeSeed(); //生成随机数种子
    // 1. 建立子进程并建立和子进程通信的信道, 有bug的,但是不影响我们后面编写
    // 1.1 加载方法表
    std::vector<func_t> funcMap;
    loadTaskFunc(&funcMap);
    // 1.2 创建子进程,并且维护好父子通信信道
    std::vector<subEp> subs;
    createSubProcess(&subs, funcMap);

    // 2. 走到这里就是父进程, 控制子进程,负载均衡的向子进程发送命令码
    int taskCnt = 3; // 0: 永远进行,>0: 父进程发送几次
    loadBlanceContrl(subs, funcMap, taskCnt);

    // 3. 回收子进程信息
    waitProcess(subs);

    return 0;
}

在这里插入图片描述


五、命名管道

1.命名管道概述

  • 管道应用的一个限制就是只能在具有共同祖先(具有亲缘关系)的进程间通信。
  • 如果我们想在不相关(不具有亲缘关系)的进程之间交换数据,可以使用FIFO文件来做这项工作,它经常被称为命名管道
  • 命名管道是一种特殊类型的文件。

2.创建一个命名管道

(1)命令行创建命名管道

$ mkfifo filename

在这里插入图片描述
在这里插入图片描述

———— 我是一条知识分割线 ————

(2)程序创建命名管道

命名管道也可以从程序里创建,相关函数有:

#include <sys/types.h>
#include <sys/stat.h>
int mkfifo(const char *filename,mode_t mode);

3.匿名管道与命名管道的区别

  • 匿名管道由pipe函数创建并打开。它通过文件指针的方式,让子进程继承父进程指向的文件,再通过文件描述符的方式看到同一份资源。
  • 命名管道由mkfifo函数创建,打开用open。它通过让不同进程看到同一个路径标定的文件,是它们能看到同一份资源。
  • FIFO(命名管道)与pipe(匿名管道)之间唯一的区别在它们创建与打开的方式不同,一但这些工作完 成之后,它们具有相同的语义。

4.命名管道的打开规则

  • 只有读端打开FIFO,写端没打开:阻塞直到有相应进程为写而打开该FIFO。
  • 只有写端打开FIFO,读端没打开:阻塞直到有相应进程为读而打开该FIFO。

六、用命名管道实现server&client通信

这里我们先要搞清楚一个概念,命名管道如何让不同的进程,看到同一份资源?

可以让不同的进程打开指定名称(路径+文件名)的同一文件

  • comm.hpp
#include <iostream>
#include <string>
#include <cstring>
#include <cerrno>
#include <cassert>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

#define NAMED_PIPE "/tmp/mypipe"

bool createFifo(const std::string& path)
{
    umask(0);
    int n = mkfifo(path.c_str(), 0600);
    if (n == 0)
        return true;
    else
    {
        std::cout << "errno: " << errno << " err string: " << strerror(errno) << std::endl;
        return false;
    }
}

void removeFifo(const std::string& path)
{
    int n = unlink(path.c_str()); //删除管道
    assert(n == 0); // debug , release 里面就没有了
    (void)n;
}
  • client.cc(写端)
#include "comm.hpp"

int main()
{
    std::cout << "client begin" << std::endl;
    int wfd = open(NAMED_PIPE, O_WRONLY);
    std::cout << "client end" << std::endl;
    if (wfd < 0) exit(1);

    //write
    char buffer[1024];
    while (true)
    {
        std::cout << "Please Say# ";
        fgets(buffer, sizeof(buffer), stdin); // abcd\n
        if (strlen(buffer) > 0) buffer[strlen(buffer) - 1] = 0; //去掉输入的\n
        ssize_t n = write(wfd, buffer, strlen(buffer));
        assert(n == strlen(buffer));
        (void)n;
    }

    close(wfd);
    return 0;
}
  • server.cc(读端)
#include "comm.hpp"

int main()
{
    bool r = createFifo(NAMED_PIPE);
    assert(r);
    (void)r;

    std::cout << "server begin" << std::endl;
    int rfd = open(NAMED_PIPE, O_RDONLY);
    std::cout << "server end" << std::endl;
    if (rfd < 0) exit(1);

    //read
    char buffer[1024];
    while (true)
    {
        ssize_t s = read(rfd, buffer, sizeof(buffer) - 1);
        if (s > 0)
        {
            buffer[s] = 0;
            std::cout << "client->server# " << buffer << std::endl;
        }
        else if (s == 0)
        {
            std::cout << "client quit, me too!" << std::endl;
            break;
        }
        else
        {
            std::cout << "err string: " << strerror(errno) << std::endl;
            break;
        }
    }

    close(rfd);

    // sleep(10);
    removeFifo(NAMED_PIPE);
    return 0;
}

注意:命名管道文件必须读端、写端都打开才能开始运行

在这里插入图片描述

注意:当我们关闭写端,读端会跟着关闭(代码实现的!)

在这里插入图片描述


结语

🌹🌹 进程间通信介绍 & 管道 的知识大概就讲到这里啦,博主后续会继续更新更多C++ 和 Linux的相关知识,干货满满,如果觉得博主写的还不错的话,希望各位小伙伴不要吝啬手中的三连哦!你们的支持是博主坚持创作的动力!💪💪

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

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

相关文章

7-1 输出倒三角

题目描述&#xff1a; 本题要求编写程序&#xff0c;输出指定的由“*”组成的倒三角图案。 输入格式: 本题目没有输入。 输出格式: 按照下列格式输出由“*”组成的倒三角图案。 * * * ** * ** **代码长度限制 16 KB 时间限制 400 ms 内存限制 64 MB 源代码&#xf…

Postman

Postman 简介下载安装 简介 Postman 是一款用于测试和开发 API&#xff08;应用程序编程接口&#xff09;的工具&#xff0c;它提供了用户友好的界面和丰富的功能&#xff0c;帮助开发者轻松地创建、测试、调试和文档化各种类型的 API。无论是在构建 Web 应用、移动应用还是其…

fork函数和exec族函数的结合使用 的案例

首先回顾之前所讲&#xff0c;在说明“为什么要创建进程”的时候&#xff0c;提到过以下两个原因&#xff1a; 其中第一个原因很好理解&#xff0c;而第二个原因就包含了上节所讲的exec族函数的知识点&#xff0c;并且不管是之前的博文还是上节的exec&#xff0c;都提到了一点“…

数据结构笔记--二叉树经典高频题

1--二叉树的最近公共祖先 主要思路&#xff1a; 最近祖先只有两种情况&#xff1a;① 自底向上&#xff0c;当两个目的结点分别在当前结点的左右子树时&#xff0c;当前结点为两个目的结点的最近祖先&#xff1b;② 最近祖先与其中一个目的结点相同&#xff0c;则另一个目的结点…

【C++】内存管理与模板

目录 一、内存管理 1.new与delete基本用法 (1) 内置类型 (2) 自定义类型 2.new, delete与malloc, free对比 (1) 内置类型 (2) 自定义类型 (3)综合特点 3.new与delete的底层实现 4. 定位new表达式 二、模板 1.引入机制 2. 基本使用 (1) 函数模板 ①概念&#xff1a…

【C语言】每日一题---1

大家好&#xff0c;我是苏貝&#xff0c;本篇博客是系列博客每日一题的第一篇&#xff0c;本系列的题都不会太难&#xff0c;如果大家对这种系列的博客感兴趣的话&#xff0c;可以给我一个赞&#x1f44d;吗&#xff0c;感谢❤️ 下面代码的结果是&#xff1a; #include <…

203、仿真-基于51单片机6自由度机械手金属液体控制报警Proteus仿真设计(程序+Proteus仿真+配套资料等)

毕设帮助、开题指导、技术解答(有偿)见文未 目录 一、硬件设计 二、设计功能 三、Proteus仿真图 四、程序源码 资料包括&#xff1a; 需要完整的资料可以点击下面的名片加下我&#xff0c;找我要资源压缩包的百度网盘下载地址及提取码。 方案选择 单片机的选择 方案一&a…

2023国赛数学建模A题B题C题D题E题思路分析 2023全国大学生数学建模思路

文章目录 0 赛题思路1 竞赛信息2 竞赛时间3 建模常见问题类型3.1 分类问题3.2 优化问题3.3 预测问题3.4 评价问题 4 建模资料 0 赛题思路 &#xff08;赛题出来以后第一时间在CSDN分享&#xff09; https://blog.csdn.net/dc_sinor?typeblog 1 竞赛信息 全国大学生数学建模…

前端开发:数组对象判断重复的方法详解

前言 在前端开发过程中,关于数据处理是非常常用的操作,尤其是通过算法处理从后端获取的数据甚为重要。而且在前端开发中,两大类型的数据处理是必备的:数组和对象。与其说是数据处理,不如说是数组和对象的处理。实际开发中,关于数组数据的处理所占比例更高,尤其是涉及到表…

uniapp项目如何运行在微信小程序模拟器上

在HbuilderX中的小程序写完后自己一定要保存&#xff0c;否则会出不来效果 那么怎么让uniapp项目运行在微信小程序开发工具中呢 1 在hbuilderx中点击运行到小程序模拟器 2 然后在项目目录中会生成一个文件夹 在微信小程序开发软件中的工具>安全设置>打开端口 或者在微…

使用sqlplus连接oracle,提示ORA-01034和ORA-27101

具体内容如下 PL/SQL Developer 处 登录时 终端处 登录时 ERROR: ORA-01034: ORACLE not available ORA-27101: shared memory realm does not exist Process ID: 0 Session ID: 0 Serial number: 0 解决方法是执行以下命令 sqlplus /nolog conn / as sysdba startup …

ECS服务器安装docker

​ 为了安装并配置 Docker &#xff0c;你的系统必须满足下列最低要求&#xff1a; 64 位 Linux 或 Windows 系统 如果使用 Linux &#xff0c;内核版本必须不低于 3.10 能够使用 sudo 权限的用户 在你系统 BIOS 上启用了 VT&#xff08;虚拟化技术&#xff09;支持 on your s…

软件测试基础篇——Linux

1、Linux系统的特征 开源免费&#xff1a; 开源&#xff1a;开放源代码&#xff0c;指的是底层的源代码是可以开放出来&#xff0c;给相关的开发者&#xff0c;根据实际的需求做出修改的。 免费&#xff1a;不花钱&#xff0c;自由传播。 ​ Linux是一种免费使用和自由传播的…

【Tomcat】(Tomcat 下载Tomcat 启动Tomcat 简单部署 基于Tomcat进行网站后端开发)

文章目录 Tomcat下载Tomcat启动Tomcat简单部署 基于Tomcat进行网站后端开发 Tomcat Tomcat 是一个 HTTP 服务器.HTTP 协议就是 HTTP 客户端和 HTTP 服务器之间的交互数据的格式. HTTP 服务器我们可以通过 Java Socket 来实现. 而 Tomcat 就是基于 Java 实现的一个开源免费,也是…

【云原生】微内核的分布式操作系统 Kubernetes

微内核的分布式操作系统 Kubernetes 如今&#xff0c;Kubernetes 已经成为分布式集群管理系统和公有云 / 私有云的事实标准。实际上&#xff0c;Kubernetes 是一个分布式操作系统&#xff0c;它是 Google 在分布式操作系统领域十余年工程经验和智慧的结晶&#xff0c;而 Google…

实时语义分割网络 BiSeNet 训练自定义数据集

语义分割是一种将标签分配给每个像素的技术&#xff0c;广泛应用于场景理解、自动驾驶、人机交互、视频监控等领域。随着卷积神经网络的不断发展&#xff0c;研究人员提出了基于全卷积网络的语义分割算法&#xff0c;这些算法在语义分割任务中表现出良好的性能。 论文地址&…

IDEA设置Maven自动编译model

IDEA设置Maven自动编译model 项目工程结构IDEA maven设置 项目工程结构 假设我们的项目结构是下图这样&#xff0c;也就是一个父工程下包含多个子模块&#xff0c;其中dubbo-01-api是公共模块&#xff0c;其它两个模块要想使用必须在pom文件中引入。 本地开发要想不会报错&am…

T113-S3-调试debug串口修改

目录 前言 一、原理图示意 二、设备树文件配置 三、系统配置文件修改 四、调试问题 总结 前言 在嵌入式系统开发过程中&#xff0c;Debug串口是一个不可或缺的工具&#xff0c;用于输出调试信息、观察系统运行状态以及进行错误排查。T113-S3开发板作为一款功能强大的嵌入式…

51单片机(普中HC6800-EM3 V3.0)实验例程软件分析 实验五 继电器

目录 前言 一、原理图及知识点介绍 1.1、继电器原理图&#xff1a; 二、代码分析 前言 第一个实验&#xff1a; 51单片机&#xff08;普中HC6800-EM3 V3.0&#xff09;实验例程软件分析 实验一 点亮第一个LED_ManGo CHEN的博客-CSDN博客 第二个实验&#xff1a;51单片机&am…

软件测试基础篇——Shell

1、 shell概述 脚本&#xff1a;也是属于文本文件/文本文档&#xff0c;除了读和写之外&#xff0c;还可以直接被执行/运行&#xff0c;一句话总结&#xff1a;一个可以直接被执行(运行)的文件/文档&#xff0c;被称为“脚本” shell脚本&#xff1a;利用shell技术编写出来的一…