进程间通信(1)——管道
1. 进程间通信简介
进程间通信(Inter-Process Communication,IPC)是指不同进程之间交换数据的机制。由于进程具有独立的地址空间,它们无法直接访问彼此的数据,因此需要IPC机制来实现信息共享、数据传递或同步操作。
1.1 进程间通信的目的
数据交换:不同进程之间共享数据,提高系统的协作能力。
资源共享:多个进程可以访问共享内存或文件,提高资源利用率。
事件通知:当一个进程发生某种事件时,能够通知其他进程进行响应。
进程同步:有些进程希望完全控制另⼀个进程的执行(如Debug进程),此时控制进程希望能够拦截另⼀个进程的所有陷⼊和异常,并能够及时知道它的状态改变。
负载均衡:多个进程可以分担任务,提高系统性能。
1.2 进程间通信的分类
System V IPC 和 POSIX IPC 与管道(Pipe)是同级别的进程间通信(IPC)机制,但它们的使用方式和适用场景不同。管道属于 IPC 机制的一种,而 System V IPC 和 POSIX IPC 提供了更丰富的通信手段,如消息队列、共享内存和信号量。
1.21 管道(Pipe)
管道是一种 基于数据流的 IPC 机制,主要用于进程间的 单向通信,分为:
(1)匿名管道(Anonymous Pipe):只能用于父子进程之间的通信,使用 pipe() 创建。
(2)命名管道(FIFO, Named Pipe):允许无亲缘关系的进程通信,使用 mkfifo() 创建。
特点:
- 适用于简单的数据流通信。
- 有序、先进先出(FIFO) 方式传输数据。
- 单向通信,如果需要双向通信,需要创建两个管道。
1.2.2 System V IPC
System V IPC 提供了三种主要的进程间通信方式:
(1)消息队列(Message Queue):进程通过发送和接收消息进行通信,类似邮件系统。
(2)共享内存(Shared Memory):多个进程可以访问同一块内存区域,比管道和消息队列快,但需要同步机制。
(3)信号量(Semaphore):用于进程同步,通常用于控制对共享资源的访问。
特点:
- 需要手动管理 IPC 资源(创建、删除)。
- 适用于 长期运行的进程,如数据库、后台服务。
- 比管道更强大,但 API 更复杂。
1.2.3 POSIX IPC
POSIX IPC 是 System V IPC 的改进版,提供:
(1)POSIX 消息队列:类似 System V 消息队列,但支持非阻塞模式,管理更方便。
(2)POSIX 共享内存:基于文件系统,使用 shm_open() 进行管理。
(3)POSIX 信号量:比 System V 信号量更简单,支持命名信号量和无名信号量。
特点:
- 资源可以自动释放,避免 System V IPC 的手动管理问题。
- 更现代化,适用于 Linux、macOS 等系统。
- API 更易用,推荐用于新开发的 Linux 应用。
管道 vs. System V IPC vs. POSIX IPC
| 机制 | 适用场景 | 亲缘关系限制 | 数据传输方式 | 速度 | 是否需要同步 |
|---|---|---|---|---|---|
| 匿名管道 | 父子进程通信 | 需要父子关系 | 字节流 | 慢 | 不需要 |
| 命名管道 | 任意进程通信 | 无亲缘关系要求 | 字节流 | 慢 | 不需要 |
| System V 消息队列 | 结构化数据传输 | 无亲缘关系要求 | 消息(队列方式) | 中等 | 不需要 |
| System V 共享内存 | 进程间高效共享数据 | 无亲缘关系要求 | 共享内存 | 最快 | 需要 |
| System V 信号量 | 进程同步 | 无亲缘关系要求 | 计数器 | 快 | 需要 |
| POSIX 消息队列 | 结构化数据传输 | 无亲缘关系要求 | 消息(队列方式) | 中等 | 不需要 |
| POSIX 共享内存 | 进程间高效共享数据 | 无亲缘关系要求 | 共享内存 | 最快 | 需要 |
| POSIX 信号量 | 进程同步 | 无亲缘关系要求 | 计数器 | 快 | 需要 |
总结
管道(Pipe)和 System V / POSIX IPC 都是 IPC 机制的一种,但它们适用于不同场景:
- 管道适用于简单的进程间数据流通信,特别是父子进程之间的通信。
- System V 和 POSIX IPC 适用于更复杂的进程通信需求,比如:
- 消息队列 适合结构化数据传输(比管道更灵活)。
- 共享内存 适合高效数据共享(比管道快)。
- 信号量 适合进程同步(配合共享内存使用)。
- POSIX IPC 是 System V IPC 的改进版,API 更简洁,资源管理更方便,推荐在现代 Linux 开发中使用。
2. 管道
管道是进程间通信(IPC)的一种方式,它允许一个进程将数据传输给另一个进程。管道在类Unix操作系统中尤其重要。管道有两种类型:匿名管道和命名管道。
-
匿名管道:匿名管道通常用于父子进程或兄弟进程之间的通信,它没有名称,因此只能在创建它的进程间使用。数据只能在一个方向上传递,从管道的一端写入数据,另一端读取数据。匿名管道的生命周期与父进程相关联。
-
命名管道(FIFO):命名管道有一个特定的名称,因此可以在不同的进程之间进行通信,而不仅限于父子进程。命名管道在文件系统中表现为一个特殊的文件,可以通过路径访问。这使得它可以跨进程、跨终端进行通信。
管道的工作原理是通过内核缓冲区进行数据传输,内核会确保进程间的数据不会相互干扰。管道是半双工的(数据只能单向流动),但通过创建两个管道可以实现全双工通信。
管道的优点是简单、快速,缺点是只能在有限的场景中使用,如通信双方需要具有亲缘关系(匿名管道)或访问权限(命名管道)。
3. 匿名管道
#include <unistd.h>
功能:创建⼀⽆名管道
原型
int pipe(int fd[2]);
参数
fd:⽂件描述符数组,其中fd[0]表⽰读端, fd[1]表⽰写端
返回值:成功返回0,失败返回错误代码
3.1 多视角理解管道
用fork来共享管道原理

#include <iostream>
#include <unistd.h>
#include <cstdio>
#include <cstring>
#include <sys/types.h>
#include <sys/wait.h>void ChildWrite(int wfd)
{char buffer[1024];int cnt = 0;while(true){snprintf(buffer, sizeof(buffer), "I am child, pid: %d, cnt: %d", getpid(), cnt++);write(wfd, buffer, strlen(buffer));sleep(1);}
}
void FatherRead(int rfd)
{char buffer[1024];while(true){buffer[0] = 0; int n = read(rfd, buffer, sizeof(buffer) - 1);if(n > 0){buffer[n] = 0;std::cout << "child say" << buffer << std::endl;}}
}int main()
{int fds[2] = {0};int n = pipe(fds);if(n < 0){std::cerr << "pipe error" << std::endl;}std::cout << "fds[0]:" << fds[0] << std::endl;std::cout << "fds[1]:" << fds[1] << std::endl;//创建子进程pid_t id = fork();if(id == 0){close(fds[0]);ChildWrite(fds[1]);close(fds[1]);exit(0);}close(fds[1]);FatherRead(fds[0]);close(fds[0]);wait(NULL);return 0;
}

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

站在内核角度-管道本质

3.2 管道的同步机制
(1)写慢,读块
读端阻塞进程(等写)
#include <iostream>
#include <unistd.h>
#include <cstdio>
#include <cstring>
#include <sys/types.h>
#include <sys/wait.h>void ChildWrite(int wfd)
{char buffer[1024];int cnt = 0;while(true){snprintf(buffer, sizeof(buffer), "I am child, pid: %d, cnt: %d", getpid(), cnt++);write(wfd, buffer, strlen(buffer));}
}
void FatherRead(int rfd)
{char buffer[1024];while(true){buffer[0] = 0; int n = read(rfd, buffer, sizeof(buffer) - 1);if(n > 0){buffer[n] = 0;std::cout << "child say" << buffer << std::endl;}sleep(1);}
}int main()
{int fds[2] = {0};int n = pipe(fds);if(n < 0){std::cerr << "pipe error" << std::endl;}std::cout << "fds[0]:" << fds[0] << std::endl;std::cout << "fds[1]:" << fds[1] << std::endl;//创建子进程pid_t id = fork();if(id == 0){close(fds[0]);ChildWrite(fds[1]);close(fds[1]);exit(0);}close(fds[1]);FatherRead(fds[0]);close(fds[0]);wait(NULL);return 0;
}

(2)写快, 读慢
缓冲区满了的时候,写端要阻塞等待读端
void ChildWrite(int wfd)
{char buffer[1024];int cnt = 0;while(true){snprintf(buffer, sizeof(buffer), "I am child, pid: %d, cnt: %d", getpid(), cnt++);write(wfd, buffer, strlen(buffer));sleep(1);}
}
void FatherRead(int rfd)
{char buffer[1024];while(true){buffer[0] = 0; int n = read(rfd, buffer, sizeof(buffer) - 1);if(n > 0){buffer[n] = 0;std::cout << "child say" << buffer << std::endl;}}
}
(3)写关, 继续读
read读到返回值为0, 表示文件结尾
void ChildWrite(int wfd)
{char buffer[1024];int cnt = 0;for (int i = 0; i < 5; ++i){snprintf(buffer, sizeof(buffer), "I am child, pid: %d, cnt: %d", getpid(), cnt++);write(wfd, buffer, strlen(buffer));sleep(1);}// 关闭写端close(wfd);
}
void FatherRead(int rfd)
{char buffer[1024];while(true){buffer[0] = 0; int n = read(rfd, buffer, sizeof(buffer) - 1);if(n > 0){buffer[n] = 0;std::cout << "child say" << buffer << std::endl;}else if (n == 0){std::cout << "Reached end of file (write end closed)" << std::endl;break;}else{std::cerr << "read error" << std::endl;break;}}
}

(4)读关,继续写
没有任何意义,OS不做没有意义的事,会杀掉进程
void ChildWrite(int wfd)
{char buffer[1024];int cnt = 0;while(true){snprintf(buffer, sizeof(buffer), "I am child, pid: %d, cnt: %d", getpid(), cnt++);write(wfd, buffer, strlen(buffer));sleep(1);}// 关闭写端close(wfd);
}
void FatherRead(int rfd)
{char buffer[1024];for(int i = 0; i < 3; i++){buffer[0] = 0; int n = read(rfd, buffer, sizeof(buffer) - 1);if(n > 0){buffer[n] = 0;std::cout << "child say" << buffer << std::endl;}}close(rfd);
}

总结:
管道(Pipe)在进程间通信(IPC)中是一种常见的机制,其读写规则如下:
当没有数据可读时:
O_NONBLOCK 关闭:read 调用会阻塞,即进程暂停执行,直到有数据可读。
O_NONBLOCK 开启:read 调用返回 -1,errno 值为 EAGAIN。
当管道满时:
O_NONBLOCK 关闭:write 调用会阻塞,直到有进程读取数据腾出空间。
O_NONBLOCK 开启:write 调用返回 -1,errno 值为 EAGAIN。
如果所有管道写端对应的文件描述符被关闭,则 read 返回 0,表示读到文件结尾(EOF)。
如果所有管道读端对应的文件描述符被关闭,则 write 操作会产生 SIGPIPE 信号,进而可能导致 write 进程退出。
当写入的数据量不大于 PIPE_BUF 时,Linux 保证写入的原子性,即写入的数据不会与其他进程的写入操作交错。
当写入的数据量大于 PIPE_BUF 时,Linux 不再保证写入的原子性,可能会发生数据交错。
3.3 基于匿名管道-进程池
管道的容量


ProcessPool.hpp创建进程池处理任务
#ifndef __PROCESS__POOL__HPP__
#define __PROCESS__POOL__HPP__#include <iostream>
#include <cstdlib>
#include <vector>
#include <string>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>#include "Task.hpp"class Channel
{
public:Channel(int fd, pid_t id) : _wfd(fd), _subid(id){_name = "channel" + std::to_string(_wfd) + "-" + std::to_string(_subid);}void Send(int code){int n = write(_wfd, &code, sizeof(code));//void(n);}~Channel() {}int getWfd() { return _wfd; }pid_t getSubid() { return _subid; }std::string getName() { return _name; }void Close(){close(_wfd);}void Wait(){pid_t rid = waitpid(_subid, nullptr, 0);(void)rid;}private:int _wfd;pid_t _subid;std::string _name;
};
class ChannelManager
{
public:ChannelManager() :_next(0){}~ChannelManager() {}void InsertChannel(int wfd, pid_t subid){_channels.emplace_back(wfd, subid);}Channel &Select(){auto &c = _channels[_next];_next++;_next %= _channels.size();return c;}void Printchannel(){for(auto &channel : _channels){std::cout << channel.getName() << std::endl;}}void Closechannel(){for(int i = _channels.size() - 1; i >= 0; i--){_channels[i].Close();std::cout << "关闭:" << _channels[i].getName() << std::endl;_channels[i].Wait();std::cout << "回收:" << _channels[i].getName() << std::endl;}}
private:std::vector<Channel> _channels;int _next;
};const int gdefaultnum = 5;class ProcessPool
{
public:ProcessPool(int num) : _process_num(num){_tm.Register(PrintLog);_tm.Register(Download);_tm.Register(Upload);}~ProcessPool() {}void Work(int rfd){while(true){int code = 0;ssize_t n = read(rfd, &code, sizeof(code));if(n > 0){if(n != sizeof(code)) continue;std::cout << "子进程[" << getpid() << "]收到一个任务码: " << code << std::endl;}else if(n == 0){std::cout << "子进程退出" << std::endl;break;}else {std::cout << "读取错误" << std::endl;break;}}}bool Create(){for (int i = 0; i < _process_num; i++){ // 1.创建管道int pipefd[2] = {0};int n = pipe(pipefd);if (n < 0)return false;// 2. 创建见子进程pid_t subid = fork();if (subid < 0)return false;else if (subid == 0){// 3. 关闭写进程// 子进程close(pipefd[1]);Work(pipefd[0]);close(pipefd[0]);exit(0);}else{// 3. 关闭读进程// 父进程close(pipefd[0]);_cm.InsertChannel(pipefd[1], subid);}}return true;}void Debug(){_cm.Printchannel();}void Run(){//1. 选择一个任务int taskcode = _tm.Code();//2. 选择一个信道,负载均衡的选择一个子进程,完成任务Channel &c = _cm.Select();std::cout << "选择了一个子进程:" << c.getName() << std::endl;//2. 发送任务c.Send(taskcode);std::cout << "发送了一个任务码:" << taskcode << std::endl;}void Stop(){_cm.Closechannel();}
private:ChannelManager _cm;int _process_num;TaskManager _tm;
};#endif
Task.hpp 创建进程池处理任务
#pragma once#include <iostream>
#include <vector>
#include <ctime>typedef void (*task_t)();debug/
void PrintLog()
{std::cout << "我是一个打印日志的任务" << std::endl;
}void Download()
{std::cout << "我是一个下载的任务" << std::endl;
}void Upload()
{std::cout << "我是一个上传的任务" << std::endl;
}
//class TaskManager
{
public:TaskManager(){srand(time(nullptr));}void Register(task_t t){_tasks.push_back(t);}int Code(){return rand() % _tasks.size();}void Execute(int code){if(code >= 0 && code < _tasks.size()){_tasks[code]();}}~TaskManager(){}
private:std::vector<task_t> _tasks;
};
Main.cc
#include "ProcessPool.hpp"int main()
{ProcessPool pp(gdefaultnum);pp.Create();//pp.Debug();//int task = 0;int cnt = 10; while(cnt--){std::cout << cnt << std::endl;pp.Run();sleep(1);}pp.Stop();return 0;
}
3.4 管道特点
(1)管道 只能用于具有共同祖先的进程(即具有亲缘关系的进程)之间的通信。通常,一个管道由一个进程创建,然后该进程调用 fork,此后父、子进程之间就可以通过该管道进行通信。
(2)管道提供 流式服务,数据按顺序传输,适用于字节流通信。
(3)管道的 生命周期随进程结束,通常进程退出后,管道会被释放。
(4)内核会对管道操作进行同步与互斥,保证数据读写的正确性。例如,多个进程写入时,写入不大于 PIPE_BUF 的数据会保持原子性。
(5)管道是 半双工的,即数据只能单向流动。如果需要双向通信,需要建立两个管道,分别用于两个方向的数据传输。

4. 命名管道
4.1 mkfifo
mkfifo 用于创建 命名管道(FIFO,First In First Out)。与普通管道(匿名管道)不同,命名管道可以用于 不具有亲缘关系的进程 之间的通信,因为它存在于文件系统中,可以由多个进程打开进行读写。



4.2 实现进程间通信
//comm.cpp
#pragma once#include <iostream>
#include <string>#define PATH "."
#define FIFONAME "fifo"#define PATH "."
#define FILENAME "fifo"#define ERR_EXIT(m) \do \{ \perror(m); \exit(EXIT_FAILURE); \} while (0)class NameFifo
{
public:NameFifo(const std::string &path, const std::string &name) : _path(path), _name(name){_fifoname = _path + "/" + _name;int n = mkfifo(_fifoname.c_str(), 0666);if(n == - 1){std::cerr << "mkfifo error" << std::endl;}std::cout << "mkfifo success" << std::endl;}~NameFifo(){int n = unlink(_fifoname.c_str());if(n == 0){std::cout << "remove FIFO_FILE success" << std::endl;}else{std::cerr << "remove FIFO_FILE failed" << std::endl;}}
private:std::string _path;std::string _name;std::string _fifoname;
};
class Fileoper
{
public:Fileoper(const std::string &path, const std::string &name) : _path(path), _name(name), _fd(-1){_fifoname = _path + "/" + _name;}void OpenForRead(){//打开,write方法中没有执行open的时候,就要在open内部进行阻塞_fd = open(_fifoname.c_str(), O_RDONLY);if(_fd < 0){std::cerr << "open error" << std:: endl;return;}std::cout << "open success" << std::endl;}void OpenForWrite(){_fd = open(_fifoname.c_str(), O_WRONLY);if(_fd < 0){std::cerr << "open fifo cerr" << std::endl;return;}std::cout << "open fifo success" << std::endl;}void Write(){std::string message;int cnt = 1;pid_t id = getpid();while(true){std::cout << "Please Enter# ";std::getline(std::cin, message);message += (", message number: " + std::to_string(cnt++) + ", [" + std::to_string(id) + "]");write(_fd, message.c_str(), message.size()); }}void Read(){while(true){char buffer[1024];int number = read(_fd, buffer, sizeof(buffer) - 1);if(number > 0){buffer[number] = 0;std::cout << "Client say# " << buffer << std::endl; }else if(number == 0){std::cout << "client quit" << number <<std::endl;break;}else{std::cerr << "client error" << std::endl;break;}}}void Close(){if(_fd > 0) close(_fd);}~Fileoper(){}
private:std::string _path;std::string _name;std::string _fifoname;int _fd;
};
//server.cc
#include <iostream>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include "comm.hpp"int main()
{//创建管道文件NameFifo fifo(PATH, FIFONAME);//文件操作Fileoper readfile(PATH, FIFONAME);readfile.OpenForRead();readfile.Read();readfile.Close();return 0;
}
//client.cc
#include <iostream>
#include <string>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include "comm.hpp"int main()
{Fileoper writefifo(PATH, FIFONAME);writefifo.OpenForWrite();writefifo.Write();writefifo.Close();return 0;
}
//Makefile
.PHONY:all
all:client serverclient::client.ccg++ -o $@ $^ -std=c++11
server:server.ccg++ -o $@ $^ -std=c++11.PHONY:clean
clean:rm client server

相关文章:
进程间通信(1)——管道
1. 进程间通信简介 进程间通信(Inter-Process Communication,IPC)是指不同进程之间交换数据的机制。由于进程具有独立的地址空间,它们无法直接访问彼此的数据,因此需要IPC机制来实现信息共享、数据传递或同步操作。 …...
python基础8 单元测试
通过前面的7个章节,作者学习了python的各项基础知识,也学习了python的编译和执行。但在实际环境上,我们需要验证我们的代码功能符合我们的设计预期,所以需要结合python的单元测试类,编写单元测试代码。 Python有一个内…...
【正点原子K210连载】第七十六章 音频FFT实验 摘自【正点原子】DNK210使用指南-CanMV版指南
第七十六章 音频FFT实验 本章将介绍CanMV下FFT的应用,通过将时域采集到的音频数据通过FFT为频域。通过本章的学习,读者将学习到CanMV下控制FFT加速器进行FFT的使用。 本章分为如下几个小节: 32.1 maix.FFT模块介绍 32.2 硬件设计 32.3 程序设…...
【杂记二】git, github, vscode等
一、前言 暂时空着... 二、git 2.1 可能的疑问 1. VSCode 项目名和 GitHub 仓库名是否需要一致? 不需要一致。 VSCode 项目名(也就是你本地的文件夹名字)和 GitHub 仓库名可以不一样。 Git 是一个分布式版本控制系统,它主要关…...
《基于Spring Boot+Vue的智慧养老系统的设计与实现》开题报告
个人主页:@大数据蟒行探索者 一、研究背景及国内外研究现状 1.研究背景 根据1982年老龄问题世界大会联合国制定的标准,如果一个国家中超过65岁的老人占全国总人口的7%以上,或者超过60岁的老人占全国总人口的10%以上,那么这个国家将被定义为“老龄化社会”[1]。 随着国…...
ModBus TCP/RTU互转(主)(从)|| Modbus主动轮询下发的工业应用 || 基于智能网关的串口服务器进行Modbus数据收发的工业应用
目录 前言 一、ModBus TCP/RTU互转(从)及应用|| 1.1 举栗子 二、ModBus TCP/RTU互转(主) 2.1 举栗子 三、ModBus 主动轮询 3.1 Modbus主动轮询原理 3.2 Modbus格式上传与下发 3.2.1.设置Modbus主动轮询指令 3.2.2 设…...
【设计模式】3W 学习法深入剖析创建型模式:原理、实战与开源框架应用(含 Java 代码)
3W 学习法总结创建型模式(附 Java 代码实战及开源框架应用) 创建型模式主要关注 对象的创建,旨在提高代码的可复用性、可扩展性和灵活性。本文采用 3W 学习法(What、Why、How),深入分析 五大创建型模式&am…...
Jobby、Quarkus 和 Spring Boot对比
Jobby、Quarkus 和 Spring Boot 是三种不同的 Java 框架,各自有不同的设计目标和适用场景。以下是对它们的详细对比: 1. 设计目标 框架设计目标Jobby轻量级的任务调度框架,专注于任务调度和执行。Quarkus面向云原生和 Kubernetes 的 Java 框…...
[代码规范]1_良好的命名规范能减轻工作负担
欢迎来到啾啾的博客🐱,一个致力于构建完善的Java程序员知识体系的博客📚,记录学习的点滴,分享工作的思考、实用的技巧,偶尔分享一些杂谈💬。 欢迎评论交流,感谢您的阅读😄…...
【HarmonyOS Next之旅】DevEco Studio使用指南(三)
目录 1 -> 一体化工程迁移 1.1 -> 自动迁移 1.2 -> 手动迁移 1.2.1 -> API 10及以上历史工程迁移 1.2.2 -> API 9历史工程迁移 1 -> 一体化工程迁移 DevEco Studio从 NEXT Developer Beta1版本开始,提供开箱即用的开发体验,将SD…...
冯・诺依曼架构深度解析
一、历史溯源:计算机科学的革命性突破 1.1 前冯・诺依曼时代 在 1940 年代之前,计算机领域呈现 "百家争鸣" 的格局: 哈佛 Mark I(1944):采用分离的指令存储与数据存储ENIAC(1946&a…...
gralloc1_perform具体在干什么
gralloc1_perform 会在特定场景下通过 ioctl 调用,执行 缓存 (cache) 管理 和 内存映射 操作,确保 CPU 和 GPU 之间的数据一致性。 📌 为什么需要对 cache 进行操作? 在 Android 系统中,CPU 和 GPU 通常共享 DDR 内存…...
安装配置Anaconda,配置VSCode
文章目录 Anaconda介绍下载Anaconda安装Anaconda换源创建一个新环境conda常用命令 VSCode环境配置 记录一下笔者收集的一些资料,不喜勿喷。 Anaconda介绍 Anaconda是一个用于科学计算的Python发行版,支持 Linux, Mac, Windows系统,提供了包管…...
【愚公系列】《高效使用DeepSeek》017-知识点思维导图生成
🌟【技术大咖愚公搬代码:全栈专家的成长之路,你关注的宝藏博主在这里!】🌟 📣开发者圈持续输出高质量干货的"愚公精神"践行者——全网百万开发者都在追更的顶级技术博主! 👉 江湖人称"愚公搬代码",用七年如一日的精神深耕技术领域,以"…...
【量化策略】网格交易策略
【量化策略】网格交易策略 🚀量化软件开通 🚀量化实战教程 技术背景与应用场景 网格交易策略是一种基于市场波动性的自动化交易方法,适用于震荡市场。它通过在预设的价格区间内设置多个买卖点,自动执行低买高卖的操作…...
C++ 语法之函数和函数指针
在上一章中 C 语法之 指针的一些应用说明-CSDN博客 我们了解了指针变量,int *p;取变量a的地址这些。 那么函数同样也有个地址,直接输出函数名就可以得到地址,如下: #include<iostream> using namespace std; void fun() …...
网络协议抓取与分析(SSL Pinning突破)
1. 网络协议逆向基础 1.1 网络协议分析流程 graph TD A[抓包环境配置] --> B[流量捕获] B --> C{协议类型} C -->|HTTP| D[明文解析] C -->|HTTPS| E[SSL Pinning突破] D --> F[参数逆向] E --> F F --> G[协议重放与模拟] 1.1.1 关键分析目标…...
蓝桥杯真题——洛谷Day13 找规律(修建灌木)、字符串(乘法表)、队列(球票)
目录 找规律 P8781 [蓝桥杯 2022 省 B] 修剪灌木 字符串 P8723 [蓝桥杯 2020 省 AB3] 乘法表 队列 P8641 [蓝桥杯 2016 国 C] 赢球票 找规律 P8781 [蓝桥杯 2022 省 B] 修剪灌木 思路:对某个特定的点来说有向前和向后的情况,即有向前再返回到该位置…...
【2025】基于Springboot + vue实现的毕业设计选题系统
项目描述 本系统包含管理员、学生、教师三个角色。 管理员角色: 用户管理:管理系统中所有用户的信息,包括添加、删除和修改用户。 配置管理:管理系统配置参数,如上传图片的路径等。 权限管理:分配和管理…...
JAVA并发编程 --- 补充内容
1 线程状态 1.1 状态介绍 当线程被创建并启动以后,它既不是一启动就进入了执行状态,也不是一直处于执行状态。线程对象在不同的时期有不同的状态。那么Java中的线程存在哪几种状态呢?Java中的线程 状态被定义在了java.lang.Thread.State枚…...
ADB三个模块介绍
ADB(Android Debug Bridge)是 Android 开发中非常重要的工具,它由 3 个主要模块 组成,分别是 ADB Client(客户端)、ADB Server(服务端) 和 ADB Daemon(守护进程ÿ…...
【ArduPilot】Windows下使用Optitrack通过MAVProxy连接无人机实现定位与导航
Windows下使用Optitrack通过MAVProxy连接无人机实现定位与导航 配置动捕系统无人机贴动捕球配置无人机参数使用MAVProxy连接Optitrack1、连接无人机3、设置跟踪刚体ID4、校正坐标系5、配置IP地址(非Loopback模式)6、启动动捕数据推流 结语 在GPS信号弱或…...
qt 图像后处理的软件一
这是一个图像后处理软件刚刚,目前功能比较单一,后续会丰富常用的功能。 目前实现的功能有 1.导入图像 2图像可中心缩放(右上角放大缩小,按钮及滚轮双重可控)。 3.图像重置功能 软件界面如下。 代码放在我的资源里…...
Ardunio 连接OLED触摸屏(SSD1106驱动 4针 IIC通信)
一、准备工作 1、硬件 UNO R3 :1套 OLED触摸屏:1套 导线诺干 2、软件 arduino 二、接线 UNO R3OLED5VVCCGNDGNDA5SCLA4SDA 脚位如下图所示: Uno R3脚位图 触摸屏脚位图 查阅显示屏的驱动规格:通常显示屏驱动芯片有SSD1306,SH110…...
深度学习 第4章 数值计算和 Deepseek 的实践
第4章 数值计算和 Deepseek 的实践 章节概述 本章主要探讨了数值计算中的关键问题,这些问题在深度学习和机器学习中尤为重要。数值计算的核心挑战在于如何在有限的计算资源和精度限制下,高效且稳定地处理连续数学问题。本章首先讨论了溢出和下溢问题&a…...
【数据分享】2000—2024年我国省市县三级逐年归一化植被指数(NDVI)数据(年最大值/Shp/Excel格式)
之前我们分享过2000-2024年我国逐年的归一化植被指数(NDVI)栅格数据,该逐年数据是取的当年月归一化植被指数(NDVI)的年最大值。(可查看之前的文章获悉详情)!该数据来源于NASA定期发布…...
HW华为流程管理体系精髓提炼华为流程运营体系(124页PPT)(文末有下载方式)
资料解读:HW华为流程管理体系精髓提炼华为流程运营体系(124页PPT) 详细资料请看本解读文章的最后内容。 华为作为全球领先的科技公司,其流程管理体系的构建与运营是其成功的关键之一。本文将从华为流程管理体系的核心理念、构建…...
Linux中管理多版本Python总结
1. pyenv 管理多个 Python 版本 pyenv 是一个流行的工具,用于管理多个 Python 版本。它可以轻松安装、切换和管理不同版本的 Python。 1.1.安装 pyenv 1.1.1安装依赖 sudo apt-get update sudo apt-get install -y make build-essential libssl-dev zlib1g-dev…...
【系统架构设计师】操作系统 - 特殊操作系统 ③ ( 微内核操作系统 | 单体内核 操作系统 | 内核态 | 用户态 | 单体内核 与 微内核 对比 )
文章目录 一、微内核操作系统1、单体内核 操作系统2、微内核操作系统 引入3、微内核操作系统 概念4、微内核操作系统 案例 二、单体内核 与 微内核 对比1、功能对比2、单体内核 优缺点3、微内核 优缺点 一、微内核操作系统 1、单体内核 操作系统 单体内核 操作系统 工作状态 : …...
OpenBMC:BmcWeb添加路由4 设置method
设置method的方式与设置权限的方式类似: //http\routing\ruleparametertraits.hpp struct RuleParameterTraits {...self_t& methods(boost::beast::http::verb method){self_t* self = static_cast<self_t*>(this);std::optional<HttpVerb> verb = httpVerb…...
