Qt 信号槽机制底层原理学习
简介
Qt的信号和槽(Signals and Slots)是Qt开发团队创造的一种特殊回调机制,提供了非常简洁易用的事件触发-函数调用机制。
原理学习
虽然上层使用简单,但底层实现机制却复杂的不得了,这里简单的学习一下大概原理。
1. 信号槽 语法
signal 声明信号,slot 声明槽函数,emit 发射信号,QObject::connect() 连接信号槽。
一个简单示例:
// 这里是 MyClass.h 头文件
// include 语句此处暂时忽略
class MyClass: public QObject
{Q_OBJECT
signals:void my_signal_test(QString text);
public:MyClass();~MyClass();public slots:void my_slot_test(QString text);
};
//
//---------------------------------------------------------------------
// 这里是 MyClass.cpp 源文件
// include 语句此处暂时忽略
//构造函数
MyClass::MyClass()
{connect(this, &MyClass::my_signal_test, this, &MyClass::my_slot_test);
}
//槽函数
void MyClass::my_slot_test(QString text)
{qDebug() << "槽函数已执行";qDebug() << text;
}
//
//---------------------------------------------------------------------
// 这里是 main.cpp 主源文件
// include 语句此处暂时忽略
int main(int argc, char *argv[])
{QCoreApplication a(argc, argv);MyClass me;emit me.my_signal_test("hello qt");return a.exec();
}
这里,简单将MyClass自己的信号绑定到自己的槽,然后在 main 中手动触发 信号。
乍一看,好像只是简单的一连串函数调用,实则大有天机。
2. 元对象处理
上面例子中,我们只是声明了 void my_signal_test(QString text);
并没有实现该函数内容,为什么能在 main 中能 emit me.my_signal_test("hello qt");
?因为我们写的Qt版C++ 不是标准C++,上面叠加了Qt自己的专属语法,像 signals、slots、emit 是Qt 专属关键字,c++编译器不能识别的,得先经过Qt专属编译器(moc:元对象编译器 meta object compiler)转换成标准C++文件,才能最终编译。
moc 会帮我们自动实现所有信号的函数体,也就是说,信号 和 槽 都是类的成员函数。信号的函数体内容是什么呢?
是大概类似下面的东东。
// SIGNAL 0
void MyClass::my_signal_test(QString _t1)
{void *_a[] = { nullptr, const_cast<void*>(reinterpret_cast<const void*>(std::addressof(_t1))) };QMetaObject::activate(this, &staticMetaObject, 0, _a);
}
简单来说,moc 会把我们的类,添加一大堆源对象信息、运行时类型信息等。并且在 信号函数中,使用QMetaObject::activate()
进行激活信号,QMetaObject::activate()
会把传过来的元信息读取,检查该信号与槽的连接情况、连接方式、接收者对象、槽函数地址,然后使用对应方式进行槽函数的调用。
我们使用 QObject::connect()
进行连接信号槽时,包含连接信息的内容 最终总得保存在一个地方吧,要不然发射信号后需要调用槽函数时,上哪查去。答案就是保存在发射者对象中,moc 会为我们的类 额外添加一个成员变量数组,来保存所有已添加的 信号-槽函数 连接信息。除了可以使用 connect 主动添加 信号槽连接,也可以使用 QObject::disconnect()
函数取消连接,参数同 connect 一样。
3. 线程
在 Qt 中,只有 Qt 对象才有资格使用 Qt 的核心机制,如信号与槽、元对象系统、事件处理等。
一个类要想成为 Qt 类,必须满足以下要求:
- 类必须直接或间接继承自 QObject
- 在类的声明中添加 Q_OBJECT 宏。
上面提到的 一大堆元对象信息,就是 moc 将 Q_OBJECT 替换来的。
每个 Qt 对象都有自己的关联线程,初始情况下,对象在哪个线程中创建的,它的关联线程就是谁。
每个Qt对象都从 QObject 继承了 QObject::thread() 函数,调用此函数,可以获取自己关联线程的指针。
很多文章博客往往都会笼统地说“某某对象在哪个线程”,这么说难免会造成误解,其实不好,在线程A中创建的对象obj,我也能有办法在线程B中使用,那我在线程B中使用此obj对象时,你能说obj在B线程中吗?,所以还是用 关联线程 这个概念好一些。
好。简单总结一下:
- Qt对象才可以使用 Qt 的核心机制,如信号与槽、元对象系统、事件处理等。
- Qt对象才有关联线程的说法。你自己随便写的 int a, pair<int,int> 等普普通通的类对象,就是一块死数据,不在整个Qt的大一统体制内。
- 另外,Qt库中,不是说前面带Q的就是Qt对象类,例如 QString 也只是简单数据类,不算上面说的 Qt 对象。成为 Qt 对象类,必须满足上面那两条要求。
现在来看看 QThread
这个类,这个是Qt的线程类,比较特殊,首先 QThread 自身是个 Qt 对象类,在哪个线程中创建了它,它的关联线程就是谁,但是吧,他是个线程类,它自己内部管理着一个新的线程。要是在这个新线程中创建了Qt对象,那么此对象关联线程是 新线程。我们可以使用从 QObject 继承来的成员函数 QObject::moveToThread()
来修改关联线程。
看个示例:
// test.h
#ifndef TEST_H
#define TEST_H
#include <QObject>
#include <QDebug>
class Test : public QObject{Q_OBJECT// OK , Test 已经成为合格的 Qt 对象类了
public:void print_hello(){qDebug() << "hello qt";}
};
#endif//----------------------------------------------------------------------------------//main.cpp#include <QCoreApplication>
#include <QThread>
#include <QDebug>
#include "test.h"int main(int argc, char *argv[])
{QCoreApplication a(argc, argv);QThread new_thread;Test mytest;qDebug() << "mytest 的关联线程: " << mytest.thread();qDebug() << "new_thread 的关联线程: " << new_thread.thread();mytest.moveToThread(&new_thread);qDebug() << "mytest 新 的关联线程: " << mytest.thread();return a.exec();
}
输出结果:
mytest 的关联线程: QThread(0x1e6b5336030, name = "Qt mainThread")
new_thread 的关联线程: QThread(0x1e6b5336030, name = "Qt mainThread")
mytest 新 的关联线程: QThread(0xacc55ff890)
可以看到,初始情况下,因为 mytest、new_thread 两个对象都是在主线程中创建的,所以它们的关联线程都是主线程,当我们使用 moveToThread() 函数转移后,被操作对象的关联线程就随之改变。
new_thread 就好比在 A 公司上班的人,自己背地里还是 B 公司的老总,如果你要问 new_thread 是哪个公司的,他会回答你是 A 公司的,另外,别人也可以跳槽到 new_thread 掌管的 B 公司。
4. 事件循环
事件循环是一个无限循环,它从操作系统或其他事件源获取事件,并将其分发给应用程序中的对象进行处理。事件循环确保应用程序能够不断地响应用户输入和其他异步事件。
上面 QCoreApplication a; return a.exec();
启动的是主事件循环,除此之外,Qt 中每个线程都有属于该线程的 一个事件循环,还是用上面的例子,new_thread 也掌管着一个事件循环,不过我们没启动,QThread::exec() 函数负责启动本线程事件循环,但是这个方法是 preotected 的。不让直接调用,需要执行 new_thread.start() 函数启动事件循环,start() 会调用 run(),run() 默认操作是调用 exec()。
事件循环不停地从事件队列中取出事件,并将其分发给目标对象处理。
- 事件过滤器:事件首先传递给事件过滤器,事件过滤器可以选择处理事件或将其传递给下一个处理器。
- 事件处理器:如果事件没有被事件过滤器处理,Qt 会调用目标对象的 event() 方法。event() 方法会根据事件类型调用特定的事件处理器方法,例如 mousePressEvent()、keyPressEvent() 等。
用一段代码展示一下(复制的别人的)
#include <QCoreApplication>
#include <QEvent>
#include <QDebug>
#include <QTimer>// 自定义事件类
class MyCustomEvent : public QEvent {
public:static const QEvent::Type MyEventType = static_cast<QEvent::Type>(QEvent::User + 1);MyCustomEvent(const QString &message): QEvent(MyEventType), message(message) {}QString getMessage() const { return message; }private:QString message;
};// 自定义对象类
class MyObject : public QObject {Q_OBJECTprotected:// 重写 event() 方法,处理自定义事件bool event(QEvent *event) override {if (event->type() == MyCustomEvent::MyEventType) {MyCustomEvent *myEvent = static_cast<MyCustomEvent*>(event);qDebug() << "Custom event received with message:" << myEvent->getMessage();return true; // 事件已处理}return QObject::event(event); // 传递给父类处理}
};// 自定义事件过滤器类
class MyEventFilter : public QObject {Q_OBJECTprotected:// 重写 eventFilter() 方法,过滤自定义事件bool eventFilter(QObject *obj, QEvent *event) override {if (event->type() == MyCustomEvent::MyEventType) {MyCustomEvent *myEvent = static_cast<MyCustomEvent*>(event);qDebug() << "Event filter caught custom event with message:" << myEvent->getMessage();return true; // 阻止事件进一步传播}return QObject::eventFilter(obj, event); // 传递给父类处理}
};int main(int argc, char *argv[])
{QCoreApplication app(argc, argv);MyObject obj;MyEventFilter filter;// 安装事件过滤器obj.installEventFilter(&filter);// 创建并发送自定义事件MyCustomEvent *event = new MyCustomEvent("Hello, Qt!");QCoreApplication::postEvent(&obj, event);// 创建一个定时器,定时退出应用程序QTimer::singleShot(5000, &app, &QCoreApplication::quit);return app.exec(); // 进入事件循环
}
5. 信号槽的连接类型
前面铺垫了那么多,终于写到这里了,全文最想写的就是这部分了。
看一下 connect 函数声明,当然 connect 有好几个重载版本,这里拿最常用的一个来说。
template <typename PointerToMemberFunction>
static QMetaObject::Connection QObject::connect (const QObject *sender, PointerToMemberFunction signal, const QObject *receiver, PointerToMemberFunction method, Qt::ConnectionType type = Qt::AutoConnection
)
五个参数分别是
1、信号发送者对象指针
2、信号成员函数指针
3、信号接收者对象指针
4、槽函数成员函数指针
5、连接类型【默认值:Qt::AutoConnection】
连接类型有以下几种:
Qt::AutoConnection
Qt::DirectConnection
Qt::QueuedConnection
Qt::BlockingQueuedConnection
Qt::UniqueConnection
Qt::SingleShotConnection
- 先说 Qt::DirectConnection ,这个是直接连接,当信号发射时,直接在信号发射线程调用接收者的槽函数。注意是信号所在线程,就是说发射信号这个动作在哪个线程发生的,槽函数就在哪个线程执行,而且是同步阻塞的,也就是在发射信号的地方,遍历发射者的保存的连接列表,从中依次调用 所有连接本信号的 槽函数(仅限Qt::DirectConnection类型的),因为是同一线程执行,所以必须等这些槽函数全部执行完返回后,发射语句 后面的逻辑才会继续执行。注意,这种连接方式和发射者(sender)的关联线程是哪个没有关系,网上一些博客,总是错误的描述成 Qt::DirectConnection 方式 是在 发射者(sender)所在线程执行槽函数。
- Qt::QueuedConnection,队列连接,当信号发射时,如果该信号的某个连接的类型是 Qt::QueuedConnection 的,则先获取 接收者(receiver)的关联线程,再获取接收者关联线程掌管的 事件循环队列,然后将 信号打包成 Qt事件,投递到 该事件循环队列中。完事返回。这就好比快递员将货物直接放你家门口,然后走人,不等着你当面签收。 当接受者关联线程的事件循环处理到此事件时,对应的槽函数才会执行。
- Qt::BlockingQueuedConnection,阻塞队列连接,和上面差不多,只不过,快递员将货物直接放你家门口后,搁那一直等着你出现,直到你出面签收完了,快递员才走。对于实际程序则是 投递完事件 后,信号线程把自己挂起,等该信号的所有槽函数执行完后,才将其唤醒,继续后面的逻辑。
- Qt::AutoConnection,自动连接,信号发射时,先检查一下 信号所在线程 和 接受者的关联线程 是否是同一线程,如果是,则使用 Qt::DirectConnection 方式,否则使用 Qt::QueuedConnection 方式。 注意:是比较 信号所在线程 和 接收者的关联线程,而不是比较 发送者的关联线程 和 接收者的关联线程。
- Qt::UniqueConnection, 去重连接,这个是和上面几种 搭配一起使用的,使用位或运算 | 组合,例如:
Qt::QueuedConnection | Qt::UniqueConnection
,不加这个的话,相同的连接可以被重复添加,例如我执行两遍
connect(this, &MyClass::my_signal_test, this, &MyClass::my_slot_test);
connect(this, &MyClass::my_signal_test, this, &MyClass::my_slot_test);
当信号发出时,槽函数会被执行两次。 要是加了 Qt::UniqueConnection,那么添加连接时,会检查相同的连接是不是已经有了,如果有了,就不添加了。
6. Qt::SingleShotConnection,单次连接,这个是和上面几种 搭配一起使用的,使用位或运算 | 组合,例如:Qt::AutoConnection | Qt::SingleShotConnection
,加上这个的话,该连接使用一次后就自动删除。效果就是,不管发射了多少次信号,槽函数只执行一次,因为第一次用完就删掉了,后面再发射信号时,没有匹配的槽函数了。除非重新手动再 connect() 添加上去。
相关文章:
Qt 信号槽机制底层原理学习
简介 Qt的信号和槽(Signals and Slots)是Qt开发团队创造的一种特殊回调机制,提供了非常简洁易用的事件触发-函数调用机制。 原理学习 虽然上层使用简单,但底层实现机制却复杂的不得了,这里简单的学习一下大概原理。…...
【Java学习笔记】包
包(package) 包的本质:实际上就是创建不同的文件夹或者目录来保存类文件 包的三大作用 区分相同名字的类 当类很多的时候可以更方便的管理类 控制访问范围 使用方法 关键字:import—>导入(引入) …...
进程的程序替换——exec系列函数的使用
目录 前言 一、替换函数 二、程序替换的本质 一些细节: 三、程序替换与环境变量间的关系 1.介绍其他参数的意义并总结 2.自定义环境变量 1)通过execcle传参全局环境变量 2)通过execcle传参自定义环境变量 3)将自定义环境变量通过p…...
【论文阅读】DETR+Deformable DETR
可变形注意力是目前transformer结构中经常使用的一种注意力机制,最近补了一下这类注意力的论文,提出可变形注意力的论文叫Deformable DETR,是在DETR的基础上进行的改进,所以顺带着把原本的DETR也看了一下。 一、DETR DETR本身是…...
ArchLinux卡死在GRUB命令行模式修复
ArchLinux卡死在GRUB命令行模式修复 文章目录 ArchLinux卡死在GRUB命令行模式修复前言一、 系统配置1.系统配置2.磁盘分区信息 二、重建GRUB引导1.插入带ArchLinux ISO的U盘,BIOS选择U盘启动并进入ArchLinux安装界面。2.挂载btrfs根目录分区3.挂载/boot分区4.进入ch…...
Docker 容器 - Dockerfile
Docker 容器 - Dockerfile 一、Dockerfile 基本结构二、Dockerfile 指令详解2.1 FROM2.2 MAINTAINER2.3 COPY2.4 ADD2.5 WORKDIR2.6 VOLUME2.7 EXPOSE2.8 ENV2.9 RUN2.10 CMD2.11 ENTRYPOINT 三、Dockerfile 创建镜像与模板3.1 Dockerfile 镜像3.2 镜像管理3.3 Dockerfile 模板…...
C++ 中二级指针的正确释放方法
C 中二级指针的正确释放 一、什么是二级指针? 简单说,二级指针就是指向指针的指针。 即: int** p;它可以指向一个 int*,而 int* 又指向一个 int 类型的变量。 常见应用场景 动态二维数组(例如 int** matrix&#x…...
解释器模式(Interpreter Pattern)
解释器模式(Interpreter Pattern) 是行为型设计模式之一,通常用于处理“语言”类问题,比如计算器、编程语言的解析等。它的核心思想是通过建立一个解释器,解析并解释由语法规则描述的语言,通常以**抽象语法…...
编译原理期末重点-个人总结——1 概论
概述 计算机语言的分类 低级语言:机器语言(唯一能被计算机执行的),汇编语言 高级语言:JAVA ,C 执行高级语言或汇编语言的步骤 高级语言程序或汇编语言程序> (通过解释 或 翻译)转…...
五一作业-day04
文章目录 1. **ps -ef是显示当前系统进程的命令,统计下当前系统一共有多少进程**2. **last命令用于显示所用用户最近1次登录情况,awk可以取出某一列,现在要取出last命令第1列并去重统计次数**3. **secure日志是用户的登录日志,过滤出secure日志中的Failed password的次数(用课堂…...
Java按字节长度截取字符串指南
在Java中,由于字符串可能包含多字节字符(如中文),直接按字节长度截取可能会导致乱码或截取不准确的问题。以下是几种按字节长度截取字符串的方法: 方法一:使用String的getBytes方法 java public static String substringByBytes(…...
[特殊字符]Git 操作实战:如何将本地项目提交到远程 Gitee 仓库
在日常开发中,我们经常需要将本地开发的项目同步到远程代码仓库中(如 GitHub、Gitee 等),以便团队协作或备份管理。本文将以 Gitee(码云) 为例,详细讲解如何将本地已有项目提交到远程仓库&#…...
【信息系统项目管理师-论文真题】2008上半年论文详解(包括解题思路和写作要点)
更多内容请见: 备考信息系统项目管理师-专栏介绍和目录 文章目录 试题1:企业级信息系统项目管理体系的建立1、写作要点2、解题思路项目管理流程和项目管理的工具试题2:项目的质量管理1、写作要点2、解题思路项目的早期阶段如何制定项目质量管理计划如何确保项目质量管理计划…...
C语言|函数的递归调用
函数的递归调用 (逐层分解,逐层合并) 自己调用自己,必须要知道什么时候停止调用,不然会造成电脑死机。 【知识点】 1 函数调用是通过栈实现的。 多个函数嵌套调用时,会按照先调用后返回的原则进行返回。 2 函数递归必须满足的两…...
QT 在圆的边界画出圆
QT 在圆的边界画出圆 QT 在圆的边界画出实心圆 在Qt中,要实现在圆的边界上绘制图形,你需要使用QPainter类来在QWidget或其子类的paintEvent中绘制。下面我将通过一个简单的例子来说明如何在Qt中绘制一个圆,并在其边界上绘制其他图形&#x…...
Guass数据库实验(数据字典设计、交叉表设计)
Assignment 2: Database Design 目录 Assignment 2: Database Design 数据库创建 新建用户bit,并创建数据库模式ass2 使用datastdui以该用户远程登陆 创建学科数据字典相关表 学科门类表 一级学科表 二级学科表 三级学科表 学科变更历史表 插入数据字典…...
算法题(139):牛可乐和魔法封印
审题: 本题需要我们将数组中包含在区间x~y之间的数据个数找到并输出 思路: 方法一:暴力解法 首先我们可以直接遍历一次数组,找到x的索引,然后再找到y的索引,并计算最终的元素个数,这里就要有O&a…...
LeetCode热题100--189.轮转数组--中等
1. 题目 给定一个整数数组 nums,将数组中的元素向右轮转 k 个位置,其中 k 是非负数。 示例 1: 输入: nums [1,2,3,4,5,6,7], k 3 输出: [5,6,7,1,2,3,4] 解释: 向右轮转 1 步: [7,1,2,3,4,5,6] 向右轮转 2 步: [6,7,1,2,3,4,5] 向右轮转 3 步: [5,6,…...
DeepSeek-Prover-V2:数学定理证明领域的新突破
前言 在人工智能飞速发展的当下,模型的迭代与创新层出不穷。 五一假期期间,DeepSeek 再次发力,推出了令人瞩目的新模型 ——DeepSeek-Prover-V2。 与大众期待的 R2 通用推理模型不同,这次 DeepSeek 将目光聚焦于数学定理证明领…...
调试——GDB、日志
调试——GDB、日志 1. gdb常用指令2. 如何生成core文件并调试?3. 如何调试正在运行的程序4. 调试多进程程序5. 调试多线程程序6. log日志 gcc编译器可以帮我们发现语法错误,但是对业务逻辑错误却无能为力。当我们想找出逻辑错误时,就需要调试…...
ARM子程序调用与返回
子程序(也叫过程、函数、方法)是一个能被调用和执行并返回到调用点那条指令的代码 段。 两个问题:如何将参数传递给子程序或从子程序中传递出来?怎么从子程序返回到调用点? 指令BSR Proc_A调用子程序Proc_A。 处理器将…...
WSL 安装 Debian 后,apt get 如何更改到国内镜像网址?
提问:Debian apt install 如何更改到国内镜像网址? 在 Debian 系统中,你可以通过修改 /etc/apt/sources.list 文件,将软件源更改为国内镜像网址,以加快软件包的下载速度。下面为你详细介绍操作步骤: 1. 备…...
SpringCloud GateWay网关
1、网关介绍 微服务网关(Microservices Gateway)是微服务架构中的核心组件,充当所有客户端请求的统一入口,负责请求的路由、过滤和聚合等操作。它是微服务与外部系统(如Web、移动端)之间的中间层࿰…...
可视化大屏开发全攻略:技术与实践指南
引言 在数字化浪潮席卷全球的当下,数据已成为企业乃至整个社会发展的核心驱动力。从繁华都市的交通管控中心,到大型企业的数据运营中枢,可视化大屏无处不在,以直观、震撼的方式展示着数据的魅力与价值。它就像是一扇通往数据世界…...
如何设计一个为QStackWidget的界面切换动画?
目录 前言 接口考虑 实现的思路 前言 笔者这段时间沉迷于给我的下位机I.MX6ULL做桌面,这里抽空更新一下QT的东西。这篇文章是跟随CCMoveWidget一样的文章,尝试分享自己如何书写这份代码的思考的过程 接口考虑 笔者不太想使用继承的方式重新写我们的…...
LeetCode 0790.多米诺和托米诺平铺:难想条件的简单动态规划
【LetMeFly】790.多米诺和托米诺平铺:难想条件的简单动态规划 力扣题目链接:https://leetcode.cn/problems/domino-and-tromino-tiling/ 有两种形状的瓷砖:一种是 2 x 1 的多米诺形,另一种是形如 "L" 的托米诺形。两种…...
模拟芯片设计中数字信号处理一些常用概念(一)
模拟芯片设计中经常用时域场景思考来解决问题,但实际上很多地方如果采用频域角度思考,解决问题更快更方便。 时域和频域的对照关系如下: a、如果时域信号是周期的,那么它的频谱就是离散的。 b、如果时域信号是非周期的,那么它的频谱就是连续的。 c、如果时域信号是离散的…...
c++进阶——AVL树主要功能的模拟实现(附带旋转操作讲解)
文章目录 AVL树的实现AVL树的概念及引入AVL树调整问题AVL树的实现AVL树的结构AVL树的插入插入的流程更新平衡因子的原则实现插入的基本框架(插入 调整平衡因子)旋转操作右单旋左单旋左右双旋右左双旋 合并旋转代码 测试部分平衡检测接口测试用例 对于其他接口的说明 AVL树的实…...
一个电商场景串联23种设计模式:创建型、结构型和行为型
理解了!你希望有一个具体的项目案例,能够涵盖所有23种设计模式,并且将它们分类为创建型、结构型和行为型。这个需求非常好,能够帮助你从实际的应用场景理解每种设计模式的用法。 为了实现这个目标,我将为你设计一个电…...
浅拷贝和深拷贝的区别
Person p1 new Person(10);Person p2 p1;p2.age 20;System.out.println(p1p2); // trueSystem.out.println(p1.age); // 20 这种做法只是复制了对象的地址,即两个变量现在是指向了同一个对象,任意一个变量,操作了对象的属性,都…...
Java开发者面试实录:微服务架构与Spring Cloud的应用
面试场景 面试官: 请介绍一下你的基本情况。 程序员: 大家好,我叫张小明,今年27岁,硕士学历,拥有5年的Java后端开发经验。主要负责基于Spring Boot开发企业级应用,以及微服务架构的设计和实现。 面试官: 好的&#…...
在Ubuntu系统中安装桌面环境
在 Ubuntu 系统中安装桌面环境可以通过包管理器 apt 或工具 tasksel 实现。以下是详细的安装方法和常见桌面环境的选择: --- ### **1. 准备系统更新** 在安装前,建议更新软件源和系统包: bash sudo apt update && sudo apt upgrade…...
多语言笔记系列:Polyglot Notebooks 中使用 xUnit 单元测试
Polyglot Notebooks 中使用 xUnit 单元测试 本文目录 Polyglot Notebooks 中使用 xUnit 单元测试[TOC](本文目录)Polgylot Notebooks 并没有直接支持单元测试框架。不能像VS里那样方便的进行单元测试。简单远行的话,可以使用下面的方案!1、引入必要的NuG…...
Cisco Packet Tracer 选项卡的使用
目录 设备Config选项卡的使用 Realtime and Simulation模式(数据包跟踪与分析) 设备Desktop选项卡的使用 设备Config选项卡的使用 Hostname NVRAM Startup Config----Load 加载 INTERFACE 点击on Save 如果,不把Running Config保存为Sta…...
杨校老师竞赛课之C++备战蓝桥杯初级组省赛
目录 1. 灯塔 题目描述 输入描述 输出描述 输入样例1 输出样例1 输入样例2 输出样例2 数据说明 2. 子区间 题目描述 输入描述 输出描述 输入样例 输出样例 数据说明 3. 染色 题目描述 输入描述 输出描述 输入样例1 输出样例1 输入样例2 输出样例2 数据…...
gcc/g++用法摘记
链接静态库 gcc main.o -L/path/to/libs -lmylib -o myprogram 【待续】...
kotlin 扩展函数
Kotlin 扩展函数的定义与使用 定义扩展函数 Kotlin 的扩展函数是一种强大的机制,允许开发者为已有的类添加额外的功能,而无需继承该类或对其进行任何修改。这种特性极大地提高了代码的灵活性和可读性。 扩展函数可以通过在函数名称前指定目标类型的接…...
机器人强化学习入门学习笔记
(1)物理引擎 物理引擎就是模拟真实世界物理规律的软件工具。它会根据你给定的物体、质量、形状、力等信息,计算这些物体在时间上的运动和相互作用。如果你设计了一个机器人,那物理引擎就是“虚拟现实世界”,让机器人在里面“活起来”,模拟它走路、抓东西、摔倒等动作。而…...
《RESTful API版本控制的哲学思辨:稳定性与创新性的终极平衡》
有效的版本控制,就如同精密仪器中的校准装置,确保API在不断升级的过程中,依然能与旧有系统无缝对接,维持整个生态的平稳运行。 不同的客户端对API的依赖程度和使用方式各不相同。有些客户端可能因为各种原因,无法及时…...
spring中spring-boot-configuration-processor的使用
spring-boot-configuration-processor 是 Spring Boot 提供的注解处理器,用于在编译阶段生成配置元数据文件(spring-configuration-metadata.json),从而优化开发体验。以下是其核心功能和使用指南: 一、核心功能 IDE 智…...
30天开发操作系统 第27天 -- LDT与库
前言 大家早上好,我们今天的第一个任务就是修复昨天晚上的那个bug。是个什么bug来着?就是用nsct命令运行的应用程序,无论是按ShiftF1还是点击窗口的“x”按钮都没有反应的那个bug啦。 我们得先来找到出问题的原因,然后才能采取对…...
std::move()详解
一、std::move()的作用和原理 本质: std::move()并不像字面意思“搬走”那些对象,而是: 将传入的对象“强制转化”为右值引用类型,从而开启“移动语义”。 在源码层面: 复制代码 template<typename T> std::…...
linux系统基本操作命令
文件和目录操作 ls:列出目录内容。 例如:ls -l 显示详细信息,ls -a 显示包括隐藏文件在内的所有文件。 cd:改变当前目录。 例如:cd /home/username 切换到指定目录。 pwd:显示当前目录的完整路径。 mk…...
python打卡day16
NumPy 数组基础 因为前天说了shap,这里涉及到数据形状尺寸问题,所以需要在这一节说清楚,后续的神经网络我们将要和他天天打交道。 知识点: numpy数组的创建:简单创建、随机创建、遍历、运算numpy数组的索引:…...
架构进阶:什么是数据架构,如何理解数据架构?(华为)
数据架构是企业架构的重要组成部分,DAMA、IBM 及国内大厂对其定义各有侧重。它包含数据资产目录、数据标准、数据模型和数据分布四个组件。数据资产目录可梳理企业数据资产,数据标准统一数据含义和规则,数据模型反映业务对象关联关系,数据分布呈现数据流动情况。数据架构是…...
基于EFISH-SCB-RK3576工控机/SAIL-RK3576核心板的KTV点歌主机技术方案(国产化替代J1900的全场景技术解析)
一、硬件架构设计 多媒体处理模块 超高清解码: RK3576 NPUGPU协同解码,支持4K60fps H.265硬解(功耗<5W),支持8路1080P视频同步预览对比J1900需外接VPU解码芯片,硬件成本降低40%,…...
Java面试深度解密:Spring Boot、Redis、日志优化、JUnit5及Kafka事务核心技术解析
模拟面试实战 面试官:请解释Spring Boot的自动配置原理?哪些关键注解参与了这一过程? xbhog:Spring Boot通过AutoConfiguration标记核心配置类,通过ConditonalOnClass和ConditionalOnMissingBean判断依赖是否存在并自…...
内存碎片深度剖析
目录 什么是内存碎片 内部碎片的解决 malloc STL二级空间配置器 外部碎片的解决 伙伴系统算法 slab分配器 什么是内存碎片 内存碎片是指在内存中存在的一些不连续的、较小的空闲内存块,这些小块内存由于太小而无法被有效地分配给程序使用,从而导…...
飞帆网页中使用 i 评论插件
https://fvi.cn/786...
DeepSeek成本控制的三重奏
知识蒸馏 使用规则引擎筛选合成数据,来替代90%的人工标注 动态精度切换:“节能模式” 根据任务复杂度自动切换FP16/INT8精度,单位token能耗低至0.0028瓦时,推理电费成本降低82% 极致压缩训练 通过以上的技术,降低训练…...