[实现Rpc] 环境搭建 | JsonCpp | Mudou库 | callBack()

目录

1. 项目介绍

2. 技术选型

3. 开发环境和环境搭建

Ubuntu-22.04环境搭建

1. 安装 wget(一般情况下默认会自带)

2. 更换国内软件源

① 备份原始 /etc/apt/sources.list 文件

② 编辑软件源文件

③ 更新软件包列表

3. 安装常用工具

3.1 安装 lrzsz 传输工具

3.2 安装编译器 gcc/g++

3.3 安装项目构建工具 make

3.4 安装调试器 gdb

3.5 安装版本控制系统 git

3.6 安装 cmake

3.7 安装 jsoncpp

4. 安装 Muduo 库

4. 第三方库使用介绍

4.1 JsonCpp库

4.1.1 Json的数据格式

4.1.2 JsonCpp介绍

4.1.3 JsonCpp使用

4.2 Muduo库

4.2.1 Muduo库是什么

4.2.2 Muduo库常见接口介绍

1. TcpServer 类

2. EventLoop 类

3. TcpConnection 类

4. TcpClient 类

5. Buffer 类

4.2.3 Muduo库的使用

1. 实现英译汉 TCP 服务器

2. 实现英译汉 TCP 客户端

3. Makefile 编译指令

运行结果

回调函数

1. 解耦网络事件处理逻辑

2. 事件驱动的核心机制

3. 提高代码复用性与可维护性

4. 非阻塞与多线程安全

5. 动态响应不同需求


1. 项目介绍

(1)

  • RPC(Remote Procedure Call)远程过程调用,是⼀种通过网络从远程计算机上请求服务,而不需要了解底层网络通信细节。
  • RPC可以使用多种网络协议进行通信, 如HTTP、TCP、UDP等, 并且在TCP/IP网络四层模型中跨越了传输层和应用层。
  • 简言之 RPC就是像调用本地方法⼀样调用远程方法。

此过程可以理解为业务处理、计算任务,更直白的说,就是程序/方法/函数等,就是像调用本地放法⼀样调用远程放法。

(2)

举个形象点谈恋爱的例子:

  • 本地过程调用:恋爱对象在你的身边, 可以随时约对象吃饭、看电影、约会等等
  • 远端过程调用:好像异地恋⼀样, 隔着千山万水, 如果想约会, 需要先和对象进行约定,在坐火车/飞机赶到约定的地点

(3)

⼀个完整RPC通信框架,大概包含以下内容:

  • 序列化协议。
  • 通信协议。
  • 连接复用。
  • 服务注册。
  • 服务发现。
  • 服务订阅和通知。
  • 负载均衡。
  • 服务监控。
  • 同步调用。
  • 异步调用。

本项目是基于C++、JsonCpp、muduo网络库实现⼀个简单、易用的RPC通信框架

实现了 同步调用、异步callback调用、异步futrue调用、服务注册/发现/上线/下线,主题发布订阅等功能设计。


2. 技术选型

(1)目前RPC的实现方案有两种:

第一种是 client和server继承公共接口:

  • 根据IDL(接口描述语言)定义公共接口。
  • 编写代码生成器根据IDL语言生成相关的C++、Java代码
  • 然后我们的客户端和服务器程序 共同向上继承公共接口即可

比如我们常用的Protobuf、json可以定义IDL接口,并生成RPC相关的代码

  • 缺点:使用pb因为生成⼀部分代码, 所以对理解不够友好
  • 如果是json定义IDL语言需要自己编写代码⽣成器难度较大一点, 暂不考虑这种方案

第二种实现⼀个远程调用接口call,然后通过传入函数名参数来调用RPC接口。本项目采用这种实现方案。

(2)网络传输的参数和返回值 怎么映射到对应的RPC 接口上?

  1. 使用protobuf的反射机制。
  2. 使用C++模板、类型萃取、函数萃取等机制。
  3. 使用更通用的类型, 比如 JSON类型, 设计好参数和返回值协议即可。

前两种技术难度和学习成本较高, 本项目使用第三种方式。

(3)网络传输怎么做?

  • 原生socket - 实现难度较⼤, 暂不考虑。
  • Boost asio库的异步通信 - 需要扩展boost库。
  • 使用muduo库,学习开发成本较低。

(4)序列化和反序列化?

  • Protobuf:可选。
  • JSON:因为项⽬需要使⽤JSON来定义函数参数和返回值,所以本项目中 直接采用JSON进行序列化和反序列化。

3. 开发环境和环境搭建

  • Linux(Centos-7.6 / Ubuntu-22.04)。
  • VSCode/Vim。
  • g++/gdb。
  • Makefile。

Ubuntu-22.04环境搭建

1. 安装 wget(一般情况下默认会自带)

在Ubuntu 22.04中,wget通常已经预安装。如果需要确保它已安装,可以运行以下命令:

sudo apt-get update
sudo apt-get install -y wget
2. 更换国内软件源

为了加速下载速度和提高稳定性,可以将官方的APT源更换为国内镜像源。

① 备份原始 /etc/apt/sources.list 文件

首先备份原来的软件源文件,以防出现问题时可以恢复:

sudo cp /etc/apt/sources.list /etc/apt/sources.list.bak
② 编辑软件源文件

接下来,使用文本编辑器打开并编辑/etc/apt/sources.list文件以添加新的源。推荐使用nano或您喜欢的其他编辑器:

sudo nano /etc/apt/sources.list

然后,在文件中添加或替换以下内容(这里使用了阿里云和清华大学的镜像源):

deb http://mirrors.aliyun.com/ubuntu/ focal main restricted universe multiverse
deb-src http://mirrors.aliyun.com/ubuntu/ focal main restricted universe multiverse
deb http://mirrors.aliyun.com/ubuntu/ focal-security main restricted universe multiverse
deb-src http://mirrors.aliyun.com/ubuntu/ focal-security main restricted universe multiverse
deb http://mirrors.aliyun.com/ubuntu/ focal-updates main restricted universe multiverse
deb-src http://mirrors.aliyun.com/ubuntu/ focal-updates main restricted universe multiverse
deb http://mirrors.aliyun.com/ubuntu/ focal-proposed main restricted universe multiverse
deb-src http://mirrors.aliyun.com/ubuntu/ focal-proposed main restricted universe multiverse
deb http://mirrors.aliyun.com/ubuntu/ focal-backports main restricted universe multiverse
deb-src http://mirrors.aliyun.com/ubuntu/ focal-backports main restricted universe multiverse# 添加清华源
deb https://mirrors.tuna.tsinghua.edu.cn/ubuntu/ focal main restricted universe multiverse
deb https://mirrors.tuna.tsinghua.edu.cn/ubuntu/ focal-updates main restricted universe multiverse
deb https://mirrors.tuna.tsinghua.edu.cn/ubuntu/ focal-backports main restricted universe multiverse
deb https://mirrors.tuna.tsinghua.edu.cn/ubuntu/ focal-security main restricted universe multiverse

请注意,这里的focal是Ubuntu 20.04 LTS的代号。对于Ubuntu 22.04,请根据实际情况更改为jammy

③ 更新软件包列表

保存更改后,通过以下命令更新APT软件包列表:

sudo apt-get update
3. 安装常用工具

现在可以安装一些常用的开发工具和库。

3.1 安装 lrzsz 传输工具

用于从远程服务器上传和下载文件:

sudo apt-get install lrzsz
3.2 安装编译器 gcc/g++

安装GNU编译器集合,包括C和C++编译器:

sudo apt-get install gcc g++
3.3 安装项目构建工具 make

用于管理和构建项目:

sudo apt-get install make
3.4 安装调试器 gdb

用于调试程序:

sudo apt-get install gdb
3.5 安装版本控制系统 git

用于代码版本管理:

sudo apt-get install git
3.6 安装 cmake

用于跨平台的项目构建系统:

sudo apt-get install cmake
3.7 安装 jsoncpp

用于解析JSON数据格式:

sudo apt-get install libjsoncpp-dev

4. 安装 Muduo 库

1. 下载源码

可以通过 git 命令从 GitHub 克隆 Muduo 的源码到本地:

# 使用 git 克隆源码
git clone https://github.com/chenshuo/muduo.git

2. 安装依赖环境

Muduo 依赖于以下库,需要通过包管理工具安装相关依赖:

# 安装 zlib 和 Boost 库
sudo apt-get install libz-dev libboost-all-dev

3. 编译与安装

执行以下步骤完成编译和安装:

  • 解压源码(如适用):
    如果下载的是压缩包(如 muduo-master.zip),需要先解压。
  • unzip muduo-master.zip
    cd muduo-master
  • 运行编译脚本
    使用 build.sh 脚本编译 Muduo 库。
  • ./build.sh
  • 安装到系统目录
    运行安装脚本将编译生成的库文件安装到系统路径。
./build.sh install

安装成功:


4. 第三方库使用介绍

4.1 JsonCpp库

4.1.1 Json的数据格式

Json简介

Json(JavaScript Object Notation)是一种轻量级的数据交换格式

  • 它采用完全独立于编程语言的文本格式来存储和表示数据。
  • 这种格式易于人阅读和编写,同时也易于机器解析和生成。

示例:学生信息表示

  • C/C++代码表示
char *name = "xx";
int age = 18;
float score[3] = {88.5, 99, 58};
  • Json 表示
{"姓名" : "xx","年龄" : 18,"成绩" : [88.5, 99, 58],"爱好" :{"书籍" : "西游记","运动" : "打篮球"}
}

Json 的数据类型

  • 对象:使用花括号 {} 括起来表示一个对象。
  • 数组:使用中括号 [] 括起来表示一个数组。
  • 字符串:使用常规双引号 "" 括起来表示一个字符串。
  • 数字:包括整形和浮点型,直接使用。

4.1.2 JsonCpp介绍

Jsoncpp库概述

Jsoncpp库主要用于实现Json格式数据的序列化和反序列化,即它可以将多个数据对象组织成Json格式字符串,并且可以从Json格式字符串解析得到多个数据对象。

⭕Json 数据对象类 Value

class Json::Value
{Value &operator=(const Value &other); // Value重载了[]和=,因此所有的赋值和获取数据都可以通过简单的方式完成Value& operator[](const std::string& key); // val["name"] = "xx";Value& operator[](const char* key);Value removeMember(const char* key); // 移除元素const Value& operator[](ArrayIndex index) const; // val["score"][0]Value& append(const Value& value); // 添加数组元素 val["score"].append(88);ArrayIndex size() const; // 获取数组元素个数 val["score"].size();std::string asString() const; // 转换为 string string name = val["name"].asString();const char* asCString() const; // 转换为 char* char *name = val["name"].asCString();Int asInt() const; // 转换为 int int age = val["age"].asInt();float asFloat() const; // 转换为 float float weight = val["weight"].asFloat();bool asBool() const; // 转换为 bool bool ok = val["ok"].asBool();
};

序列化接口

class JSON_API StreamWriter 
{virtual int write(Value const& root, std::ostream* sout) = 0;
};class JSON_API StreamWriterBuilder : public StreamWriter::Factory 
{virtual StreamWriter* newStreamWriter() const;
};

反序列化接口

class JSON_API CharReader 
{virtual bool parse(char const* beginDoc, char const* endDoc, Value* root, std::string* errs) = 0;
};class JSON_API CharReaderBuilder : public CharReader::Factory 
{virtual CharReader* newCharReader() const;
};

序列化和反序列化都是需要通过工厂类来进行实例化。


4.1.3 JsonCpp使用

代码编写示例

下面是一个简单的例子,展示了如何使用Jsoncpp库进行序列化和反序列化操作。

#include <iostream>
#include <string>
#include <jsoncpp/json/json.h>
#include <sstream>
#include <memory>// 序列化函数
bool serialize(const Json::Value &val, std::string &body)
{std::stringstream ss;Json::StreamWriterBuilder swb;std::unique_ptr<Json::StreamWriter> w(swb.newStreamWriter());bool ret = w->write(val, &ss);if (!ret) return false;body = ss.str();return true;
}// 反序列化函数
bool deserialize(const std::string &body, Json::Value &val)
{Json::CharReaderBuilder crb;std::unique_ptr<Json::CharReader> r(crb.newCharReader());std::string errs;bool ret = r->parse(body.c_str(), body.c_str() + body.size(), &val, &errs);if (!ret) {std::cout << "json unserialize failed : " << errs << std::endl;return false;}return true;
}int main()
{const char* sex = "男";const char* name = "小明";int age = 18;int score[4] = {70, 80, 90};Json::Value stu1;stu1["姓名"] = name;stu1["性别"] = sex;stu1["年龄"] = age;for (auto s : score) {stu1["分数"].append(s);}Json::Value fav;fav["书籍"] = "西游记";fav["运动"] = "打篮球";stu1["爱好"] = fav;std::string body;bool ret1 = serialize(stu1, body);std::cout << body << std::endl;std::string str = R"({"姓名":"小黑", "年龄": 19, "成绩":[32, 45, 56]})";Json::Value stu2;bool ret2 = deserialize(str, stu2);if (!ret2) return -1;std::cout << "姓名: " << stu2["姓名"].asString() << std::endl;std::cout << "年龄: " << stu2["年龄"].asInt() << std::endl;for (int i = 0; i < static_cast<int>(stu2["成绩"].size()); ++i) {std::cout << "成绩: " << stu2["成绩"][i].asFloat() << std::endl;}return 0;
}

⭕注意:对于头文件的包含调用

运行:


4.2 Muduo库

4.2.1 Muduo库是什么

Muduo是由陈硕老师开发的 基于非阻塞IO和事件驱动的C++高并发TCP网络编程库。它采用 主从Reactor模型one loop per thread 线程模型

⭕one loop per thread 的含义:

  1. 一个线程只能有一个事件循环(EventLoop),用于响应计时器和IO事件。
  2. 一个 文件描述符只能由一个线程进行读写,即一个TCP连接必须归属于某个EventLoop管理。

从而避免了,例如一万个请求就创建一万个线程的等待浪费,主从的结构也将 对于网络连接的接收和对于信息的处理 进行了拆分。


4.2.2 Muduo库常见接口介绍
1. TcpServer 类

TcpServer 用于服务端管理和维护 TCP 连接。核心接口包括:

  • 构造函数
TcpServer(EventLoop* loop, const InetAddress& listenAddr, const string& nameArg, Option option = kNoReusePort);
    • 参数:
      • loop:主EventLoop,用于事件循环。
      • listenAddr:服务器监听地址。
      • nameArg:服务器实例名称。
      • option:是否复用端口。
  • 线程设置与启动
void setThreadNum(int numThreads); // 设置IO线程数量
void start();                      // 启动服务端监听
  • 回调函数设置
void setConnectionCallback(const ConnectionCallback& cb); // 连接建立/断开回调
void setMessageCallback(const MessageCallback& cb);       // 消息接收回调
2. EventLoop 类

EventLoop 是事件循环的核心类。

  • 循环控制
void loop();  // 启动事件循环
void quit();  // 退出事件循环
  • 定时器接口
TimerId runAt(Timestamp time, TimerCallback cb);      // 在指定时间运行回调
TimerId runAfter(double delay, TimerCallback cb);     // 指定延迟后运行回调
TimerId runEvery(double interval, TimerCallback cb);  // 周期性运行回调
void cancel(TimerId timerId);                         // 取消定时器
3. TcpConnection 类

TcpConnection 负责维护单个TCP连接。

  • 核心功能
void send(const void* message, int len);  // 发送数据
void shutdown();                         // 关闭连接
  • 上下文管理
void setContext(const boost::any& context); // 设置上下文
const boost::any& getContext() const;       // 获取上下文
  • 回调函数
void setConnectionCallback(const ConnectionCallback& cb); // 设置连接回调
void setMessageCallback(const MessageCallback& cb);       // 设置消息回调
4. TcpClient 类

TcpClient 用于客户端管理 TCP 连接。

  • 核心方法
void connect();    // 发起连接
void disconnect(); // 断开连接
void stop();       // 停止客户端
  • 回调函数
void setConnectionCallback(ConnectionCallback cb); // 设置连接回调
void setMessageCallback(MessageCallback cb);       // 设置消息回调
  • 注意事项
    由于TcpClient操作是异步的,在连接建立之前尝试发送数据是不被允许的,可通过CountDownLatch进行同步。
5. Buffer 类

Buffer 用于管理数据缓冲区。

  • 核心方法
size_t readableBytes() const;  // 可读数据大小
size_t writableBytes() const;  // 可写数据大小
string retrieveAllAsString();  // 取出所有数据并转为字符串
void append(const char* data, size_t len); // 添加数据
void prepend(const void* data, size_t len); // 在缓冲区头部追加数据
  • 内部实现
    使用std::vector<char>作为缓冲区,支持动态扩容。

4.2.3 Muduo库的使用
1. 实现英译汉 TCP 服务器

代码示例

  • 核心逻辑
    • 使用TcpServer管理服务器。
    • 注册onConnectiononMessage回调函数。
    • onMessage中,通过字典查找翻译单词,并返回给客户端。

代码实现

class DictServer {
public:DictServer(int port):_server(&_baseloop, InetAddress("0.0.0.0", port), "DictServer", TcpServer::kReusePort){_server.setConnectionCallback(std::bind(&DictServer::onConnection, this, std::placeholders::_1));_server.setMessageCallback(std::bind(&DictServer::onMessage, this, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3));}void Start() {_server.start();_baseloop.loop();}private:void onConnection(const TcpConnectionPtr &conn) {if (conn->connected()) std::cout << "连接建立!\n";else std::cout << "连接断开!\n";}void onMessage(const TcpConnectionPtr &conn, Buffer *buf, Timestamp) {std::unordered_map<std::string, std::string> dict = {{"hello", "你好"}, {"world", "世界"}};std::string msg = buf->retrieveAllAsString();conn->send(dict.count(msg) ? dict[msg] : "未知单词!");}private:EventLoop _baseloop;TcpServer _server;
};int main() {DictServer server(8080);server.Start();return 0;
}
2. 实现英译汉 TCP 客户端

代码示例

  • 核心逻辑
    • 使用TcpClient连接服务器。
    • 注册回调函数onConnectiononMessage
    • 在建立连接后,通过send方法发送单词。

代码实现

class DictClient {
public:DictClient(const std::string &ip, int port):_baseloop(_loopthread.startLoop()), _latch(1), _client(_baseloop, InetAddress(ip, port), "DictClient"){_client.setConnectionCallback(std::bind(&DictClient::onConnection, this, std::placeholders::_1));_client.setMessageCallback(std::bind(&DictClient::onMessage, this, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3));_client.connect();_latch.wait();}void send(const std::string &msg) {if (_conn->connected()) _conn->send(msg);else std::cout << "连接断开,发送失败!\n";}private:void onConnection(const TcpConnectionPtr &conn) {if (conn->connected()) {_conn = conn;_latch.countDown();} else {_conn.reset();}}void onMessage(const TcpConnectionPtr &conn, Buffer *buf, Timestamp) {std::cout << buf->retrieveAllAsString() << std::endl;}private:EventLoop *_baseloop;EventLoopThread _loopthread;TcpClient _client;TcpConnectionPtr _conn;CountDownLatch _latch;
};int main() {DictClient client("127.0.0.1", 8080);std::string msg;while (std::cin >> msg) client.send(msg);return 0;
}
3. Makefile 编译指令
CFLAG= -std=c++11 -I ../build/release-install-cpp11/include/
LFLAG= -L../build/release-install-cpp11/lib -lmuduo_net -lmuduo_base -pthreadall: server clientserver: server.cppg++ $(CFLAG) $^ -o $@ $(LFLAG)client: client.cppg++ $(CFLAG) $^ -o $@ $(LFLAG)clean:rm -f server client

运行结果

通过简单输入单词实现实时翻译,服务器返回翻译结果或提示“未知单词”。

客户端

回调函数

在 Muduo 中使用回调函数(如 setConnectionCallbacksetMessageCallback)是为了实现网络库的 事件驱动编程模型,这符合非阻塞 I/O 和高并发网络编程的设计理念。

🔷 为什么要借助回调函数呢


1. 解耦网络事件处理逻辑

Muduo 库本身只负责处理网络事件的底层逻辑,比如连接建立、断开、消息接收等,但具体的业务逻辑是用户的需求(如数据如何处理、何时响应客户端等),而这些需求是 多变且不可预测的

  • 问题:如果将业务逻辑直接嵌入到 Muduo 内部,不仅会导致耦合度过高,还会限制灵活性。
  • 解决通过 回调函数,用户可以将自己的业务逻辑注册给 Muduo,在事件发生时 Muduo 自动调用注册的回调函数。

示例

  • 连接建立或断开时,用户可以通过 setConnectionCallback 注册逻辑来处理事件(如记录日志、初始化资源等)。
  • 消息到达时,通过 setMessageCallback 注册回调处理消息(如解析请求、响应客户端)。

2. 事件驱动的核心机制

Muduo 的设计基于 事件驱动模型,回调函数是这种模型的核心:

  1. 在事件循环(EventLoop)中,Muduo 会监听文件描述符(如 socket)上的事件。
  2. 当检测到事件(如新连接、可读数据等),会触发相应的回调函数。

使用回调函数的好处:

  • 异步非阻塞:Muduo 不会阻塞等待事件完成,而是注册回调函数后,立即返回继续处理其他事件。
  • 灵活性高:用户可以根据业务需要,提供任意逻辑来响应事件,而无需修改框架本身。

示例

void onConnection(const TcpConnectionPtr& conn) {if (conn->connected()) {std::cout << "新连接建立:" << conn->peerAddress().toIpPort() << std::endl;} else {std::cout << "连接断开" << std::endl;}
}

用户通过 setConnectionCallbackonConnection 注册给 TcpServer,当连接建立或断开时,TcpServer 自动调用用户定义的逻辑。


3. 提高代码复用性与可维护性

回调函数机制使得:

  • 复用性高:Muduo 的底层逻辑和用户的业务逻辑是解耦的,可以分别复用。例如,Muduo 可用于不同类型的服务器(如 HTTP 服务器、聊天服务器),而无需修改核心代码。
  • 易于维护:用户只需关心自己的回调函数,不需要深入理解网络库内部的实现。框架和业务逻辑的分离使得代码更清晰易维护。

4. 非阻塞与多线程安全

在高并发场景中,回调函数的设计可以帮助 Muduo:

  1. 避免阻塞:通过异步回调实现逻辑处理,避免阻塞事件循环线程(EventLoop)。
  2. 线程安全:Muduo 使用 one loop per thread 模型,每个线程一个 EventLoop,回调函数在绑定的线程中执行,不需要担心多线程竞态问题。

5. 动态响应不同需求

回调函数的动态绑定特性允许用户在运行时根据需求改变逻辑:

  • 用户可以在程序启动时或运行时,调用 setConnectionCallbacksetMessageCallback 设置不同的回调函数,动态调整行为。

示例

// 设置新的消息回调函数
server.setMessageCallback([](const TcpConnectionPtr& conn, Buffer* buf, Timestamp) {std::string msg = buf->retrieveAllAsString();conn->send("Echo: " + msg);
});

通过新的回调函数,程序逻辑可以轻松从英译汉服务器切换为回声服务器,而无需修改其他部分的代码。

总结

使用回调函数的原因总结如下:

  1. 解耦网络库与业务逻辑,增强灵活性。
  2. 事件驱动模型的核心机制,支持异步非阻塞操作。
  3. 提高复用性与可维护性,框架和业务逻辑分离。
  4. 适应高并发需求,避免阻塞并提供线程安全保证。
  5. 支持动态逻辑调整,满足运行时多样化需求。

回调函数是事件驱动编程中不可或缺的设计模式,Muduo 通过它实现了高性能、高并发的网络框架设计理念。

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

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

相关文章

Golang Gin系列-1:Gin 框架总体概述

本文介绍了Gin框架&#xff0c;探索了它的关键特性&#xff0c;并建立了简单入门的应用程序。在这系列教程里&#xff0c;我们会探索Gin的主要特性&#xff0c;如路由、中间件、数据库集成等&#xff0c;最终能使用Gin框架构建健壮的web应用程序。 总体概述 Gin是Go编程语言的…

游戏引擎学习第81天

仓库:https://gitee.com/mrxiao_com/2d_game_2 或许我们应该尝试在地面上添加一些绘图 在这段时间的工作中&#xff0c;讨论了如何改进地面渲染的问题。虽然之前并没有专注于渲染部分&#xff0c;因为当时主要的工作重心不在这里&#xff0c;但在实现过程中&#xff0c;发现地…

IO多路复用详解-selectpollepoll

目录 1.IO多路复用概念 2.系统调用函数 2.1select 2.1.1select函数细节 2.2基于select实现并发处理 2.2.1处理流程 2.2.2服务端通信代码 2.2.3客户端通信代码 2.3基于poll函数实现并发处理 2.3.1select与poll函数区别 2.3.2poll函数 2.3.3服务器端代码实现 2.3.4客…

IDEA下载安装

目录 IDEAWin下载安装 Mac下载安装 IDEA中基本配置&注释修改背景主题为白色修改字体大小鼠标滚轮控制字体大小控制字母大小写提示&#xff08;取消勾选&#xff09;设置自动编译&#xff08;打勾&#xff09;自动保存&#xff08;参数为1&#xff09;设定参数提示&#xff…

.Net Core微服务入门全纪录(五)——Ocelot-API网关(下)

系列文章目录 1、.Net Core微服务入门系列&#xff08;一&#xff09;——项目搭建 2、.Net Core微服务入门全纪录&#xff08;二&#xff09;——Consul-服务注册与发现&#xff08;上&#xff09; 3、.Net Core微服务入门全纪录&#xff08;三&#xff09;——Consul-服务注…

RV1126+FFMPEG推流项目(9)AI和AENC模块绑定,并且开启线程采集

前面两篇已经交代AI和AENC模块的配置&#xff0c;这篇就让这两个模块绑定起来&#xff0c;绑定的原因是&#xff0c;Aenc从Ai模块拿到采集的原始数据进行编码。 使用 RK_MPI_SYS_Bind 把 AI 节点和 AENC 进行绑定&#xff0c;其中 enModId 是模块 ID 号选择的是 RK_ID_AI、s32C…

2.5G PoE交换机 TL-SE2109P 简单开箱评测,8个2.5G电口+1个10G光口(SFP+)

TPLINK&#xff08;普联&#xff09;的万兆上联的2.5G网管交换机TL-SE2109P简单开箱测评。8个PoE 2.5G电口&#xff0c;1个万兆SFP上联口。 2.5G交换机 TL-SE2420 简单开箱评测&#xff0c;16个2.5G电口4个10G光口(SFP)&#xff1a;https://blog.zeruns.com/archives/837.html…

学成在线_内容管理模块_创建模块工程

学成在线模块工程 1.各个微服务依赖基础工程2.每个微服务都是一个前后端分离的项目3.xuecheng-plus-content&#xff1a;内容管理模块工程xuecheng-plus-content-modelxuecheng-plus-content-servicexuecheng-plus-content-api 1.各个微服务依赖基础工程 2.每个微服务都是一个前…

GCPAAS/DashBoard:完全免费的仪表盘设计,基于Vue+ElementUI+G2Plot+Echarts,开源代码,简单易用!还在等什么呢

嗨&#xff0c;大家好&#xff0c;我是小华同学&#xff0c;关注我们获得“最新、最全、最优质”开源项目和高效工作学习方法 GCPAAS/DashBoard&#xff0c;一款基于SpringBoot、MyBatisPlus、ElementUI、G2Plot、Echarts等技术栈的仪表盘设计器&#xff0c;具备仪表盘目录管理…

登录校验Cookie、Session、JWT

目录 基础知识&#xff1a;登录校验的场景 基础知识&#xff1a;会话 ​编辑方案一&#xff1a;Cookie 方案二&#xff1a;Session 方案三&#xff1a;令牌技术 JWT 令牌 基础知识&#xff1a;登录校验的场景 基础知识&#xff1a;会话 什么是会话&#xff1f;什么是会话跟…

Android BitmapShader实现狙击瞄具十字交叉线准星,Kotlin

Android BitmapShader实现狙击瞄具十字交叉线准星&#xff0c;Kotlin <?xml version"1.0" encoding"utf-8"?> <RelativeLayout xmlns:android"http://schemas.android.com/apk/res/android"xmlns:tools"http://schemas.android.…

ABP - 缓存模块(1)

ABP - 缓存模块&#xff08;1&#xff09; 1. 与 .NET Core 缓存的关系和差异2. Abp 缓存的使用2.1 常规使用2.2 非字符串类型的 Key2.3 批量操作 3. 额外功能 1. 与 .NET Core 缓存的关系和差异 ABP 框架中的缓存系统核心包是 Volo.Abp.Caching &#xff0c;而对于分布式缓存…

浅谈云计算22 | Kubernetes容器编排引擎

Kubernetes容器编排引擎 一、Kubernetes管理对象1.1 Kubernetes组件和架构1.2 主要管理对象类型 二、Kubernetes 服务2.1 服务的作用与原理2.2 服务类型 三、Kubernetes网络管理3.1 网络模型与目标3.2 网络组件3.2.1 kube-proxy3.2.2 网络插件 3.3 网络通信流程 四、Kubernetes…

如何在linux系统上完成定时开机和更新github端口的任务

任务背景 1.即使打开代理&#xff0c;有的时候github去clone比较大的文件时也会出问题。这时需要每小时更新一次github的host端口&#xff1b; 2.马上要放假&#xff0c;想远程登录在学校的台式电脑&#xff0c;但学校内网又不太好穿透。退而求其次&#xff0c;选择定时启动电…

vector迭代器的使用以及迭代器失效

一、iterator的使用注意 begin与end 遵循左闭右开的原则&#xff0c;begin 指向vector的第一个元素&#xff0c;end 指向vector的最后一个元素的往下一个位置。 rbegin 与 rend rbegin指向最后一个元素的位置&#xff0c;rend指向第一个元素的往前一个位置。 二、vector的常…

无降智o1 pro——一次特别的ChatGPT专业模式探索

这段时间和朋友们交流 ChatGPT 的使用心得&#xff0c;大家都提到一个很“神秘”的服务&#xff1a;它基于 O1 Pro 模型&#xff0c;能够在对话里一直保持相对高水平的理解和回复&#xff0c;不会突然变得“降智”。同时&#xff0c;整体使用还做了免折腾的网络设置——简单一点…

Web前端开发技术之HTMLCSS知识点总结

学习路线 一、新闻网界面1. 代码示例2. 效果展示3. 知识点总结3.1 HTML标签和字符实体3.2 超链接、颜色描述与标题元素3.3 关于图片和视频标签&#xff1a;3.4 CSS引入方式3.5 CSS选择器优先级 二、flex布局1. 代码示例2. 效果展示3. 知识点总结3.1 span标签和flex容器的区别3.…

ToDesk设置临时密码和安全密码都可以当做连接密码使用

ToDesk 在各领域办公都已经是非常常见了 为了安全 ToDesk 设置了连接密码&#xff0c;想连接 需要输入远程码和连接密码 我们刚打开 系统默认给我们用的是临时密码&#xff0c;安全性确实很强 和定时Tokey一样&#xff0c;固定时间切换。 但是 如果我们要经常连接这个电脑&a…

如何在vue中渲染markdown内容?

文章目录 引言什么是 markdown-it&#xff1f;安装 markdown-it基本用法样式失效&#xff1f;解决方法 高级配置语法高亮 效果展示 引言 在现代 Web 开发中&#xff0c;Markdown 作为一种轻量级的标记语言&#xff0c;广泛用于文档编写、内容管理以及富文本编辑器中。markdown…

openharmony应用开发快速入门

开发准备 本文档适用于OpenHarmony应用开发的初学者。通过构建一个简单的具有页面跳转/返回功能的应用&#xff08;如下图所示&#xff09;&#xff0c;快速了解工程目录的主要文件&#xff0c;熟悉OpenHarmony应用开发流程。 在开始之前&#xff0c;您需要了解有关OpenHarmon…