AceRoutine:面向嵌入式平台的零栈协程库

news2026/3/24 19:29:12
1. AceRoutine面向资源受限嵌入式平台的零栈协程库深度解析1.1 设计哲学与工程定位AceRoutine 并非传统意义上的“多线程”库而是一个严格遵循协作式调度cooperative scheduling原则、采用零栈stackless协程模型的轻量级任务管理框架。其核心设计目标直指嵌入式开发中最敏感的两大瓶颈静态内存RAM占用与上下文切换开销。在 Arduino 生态中尤其是资源捉襟见肘的 8 位 AVR 平台如 ATmega328P一个标准 FreeRTOS 任务往往需要数百字节的 RAM 作为私有栈空间而 AceRoutine 的单个协程仅消耗16 字节静态 RAMAVR或 28 字节32 位平台且不依赖任何动态内存分配。这种极致精简并非牺牲功能而是源于对嵌入式场景的深刻洞察绝大多数传感器读取、LED 控制、串口协议解析等任务并不需要复杂的函数调用栈来保存局部变量状态。它们的核心诉求是可暂停、可恢复的执行流。AceRoutine 通过 C 类封装与 GCC 特定语言扩展将协程的“状态机”逻辑内聚于Coroutine对象的成员变量中彻底规避了为每个任务分配独立栈空间的沉重负担。其本质是 ProtoThreads 思想在 Arduino C 环境下的现代化重构摒弃了原始 ProtoThreads 中晦涩难懂的 Duffs Device 技巧转而利用 GCC 的computed goto实现更清晰、更健壮的状态跳转。1.2 核心架构与类图解析AceRoutine 的架构异常精炼仅由两个核心类构成稳定基座其余均为可选增强模块。1.2.1 Coroutine 类协程的“灵魂容器”Coroutine是所有用户协程的基类它本身不包含任何业务逻辑而是一个纯粹的状态管理器。其关键成员变量定义了协程的全部“身份信息”成员变量类型说明AVR 占用32 位平台占用mStateuint16_t当前执行状态码由宏自动生成的switch语句驱动2 字节2 字节mNameconst char*协程名称指针用于调试与 Profiling2 字节4 字节mProfilerCoroutineProfiler*指向性能分析器的指针可为空2 字节4 字节mNextCoroutine*指向链表中下一个协程的指针供 Scheduler 使用2 字节4 字节总计——16 字节28 字节Coroutine类提供了一个纯虚函数int runCoroutine() override这是所有用户协程必须实现的入口点。该函数的返回值无实际意义始终为 0其存在纯粹是为了满足 C 多态性要求使CoroutineScheduler能够统一调用。1.2.2 CoroutineScheduler 类可选的“中央调度器”CoroutineScheduler是一个单例Singleton类它并非必需而是为简化多协程管理提供的便利层。其设计极具工程智慧它自身仅占用极小的固定内存AVR 仅 2 字节32 位平台仅 4 字节其调度能力完全依赖于Coroutine对象内部的mNext指针构建的单向链表。这意味着调度器的内存开销与协程数量完全解耦。// CoroutineScheduler 的核心调度循环简化版 void CoroutineScheduler::loop() { for (Coroutine* c mHead; c ! nullptr; c c-mNext) { c-runCoroutine(); // 通过虚函数表调用具体协程的 runCoroutine() } }此设计带来两个关键特性零额外 RAM 开销调度器本身不为协程列表分配任何 RAM所有链表节点即mNext指针都存储在Coroutine对象内部。编译期确定性链表结构在编译时由COROUTINE()宏自动构建运行时无任何动态链表操作杜绝了堆碎片风险。1.3 协程生命周期与宏系统消除样板代码的艺术AceRoutine 的易用性核心在于其精心设计的宏系统。这些宏并非简单的文本替换而是将 C 的面向对象特性和 C 预处理器的元编程能力结合实现了“声明即定义”的开发体验。1.3.1 COROUTINE() 宏一键生成协程实例COROUTINE(blinkLed)是最常用的宏它在单行代码中完成了三件关键工作定义一个Coroutine的子类名为blinkLed_Coroutine。定义一个全局静态实例名为blinkLed类型为blinkLed_Coroutine。自动注册到CoroutineScheduler链表如果使用了 Scheduler。其展开后的伪代码逻辑如下class blinkLed_Coroutine : public ace_routine::Coroutine { public: int runCoroutine() override { // 用户代码将被插入此处 } }; static blinkLed_Coroutine blinkLed; // 如果启用了 Scheduler则执行 blinkLed.mNext CoroutineScheduler::mHead; CoroutineScheduler::mHead blinkLed;1.3.2 COROUTINE_BEGIN/END协程体的语法糖COROUTINE_BEGIN()和COROUTINE_END()宏共同构成了协程函数体的“外壳”。它们的本质是生成一个switch(mState)语句并将用户代码包裹其中。mState变量在每次yield后被更新下次调用runCoroutine()时switch语句便能精准跳转到上次中断的位置。// 用户编写的协程体 COROUTINE(blinkLed) { COROUTINE_LOOP() { // 展开为 while(true) { ... } digitalWrite(LED, HIGH); COROUTINE_DELAY(100); // 展开为 mState __LINE__; return; case __LINE__: digitalWrite(LED, LOW); COROUTINE_DELAY(500); } }COROUTINE_DELAY(100)宏的精妙之处在于它利用了 C 的case标签和goto的组合。它首先将当前行号__LINE__存入mState然后执行return退出函数当调度器再次调用该协程时switch(mState)会直接跳转到case 上次行号从而实现了“从断点处继续执行”的效果。1.3.3 其他核心宏构建异步逻辑的积木宏功能底层机制注意事项COROUTINE_YIELD()主动让出 CPUmState __LINE__; return; case __LINE__:最基础的挂起方式COROUTINE_AWAIT(condition)等待条件成立while(!(condition)) { COROUTINE_YIELD(); }条件表达式需为纯计算避免阻塞COROUTINE_DELAY(millis)毫秒级延时基于millis()的轮询等待millis参数为uint16_t最大 65535msCOROUTINE_DELAY_MICROS(micros)微秒级延时基于micros()的轮询等待micros参数为uint16_t最大 65535μsCOROUTINE_DELAY_SECONDS(seconds)秒级延时基于millis()的长周期等待seconds参数为uint16_t最大 65535sCOROUTINE_CHANNEL_WRITE/READ通道通信调用Channel类的同步读写方法实验性功能仅支持单生产者-单消费者1.4 性能剖析为什么 AceRoutine 如此之快AceRoutine 的“微秒级”上下文切换性能是其核心竞争力其速度优势源于对底层硬件和编译器特性的极致利用。1.4.1 直接调度Direct Scheduling模式当用户在loop()中手动调用coroutine.runCoroutine()时即为直接调度模式。此时runCoroutine()是一个普通的、非虚函数的成员函数调用如果未使用 Scheduler。其执行路径极短读取mState。执行switch(mState)。跳转到对应case标签。执行用户代码直到遇到下一个COROUTINE_YIELD。在 16MHz ATmega328P 上此过程仅需约1.0 微秒。这几乎等同于一次函数调用加一次switch的开销是软件层面所能达到的理论极限。1.4.2 协程调度Coroutine Scheduling模式当使用CoroutineScheduler::loop()时由于需要遍历链表并调用虚函数性能会略有下降ATmega328P 约 5.2 微秒。但这个代价换来的是代码组织的巨大便利性。更重要的是这个“损失”是恒定的——无论你有 1 个还是 100 个协程loop()函数本身的执行时间几乎不变因为链表遍历是 O(n)而runCoroutine()的调用开销是 O(1)。1.4.3 编译器优化的关键Computed GotoAceRoutine 放弃 ANSI C 兼容性坚定选择 GCC/Clang 的computed goto扩展label语法这是其性能超越 ProtoThreads 的根本原因。Duffs Device 依赖于switch语句的 fall-through 特性导致无法在协程体内嵌套switch严重限制了代码结构。而computed goto允许直接跳转到任意标签使得COROUTINE_AWAIT()等复杂宏的实现成为可能且生成的汇编代码更为紧凑高效。1.5 内存消耗在字节级别上的精打细算对于嵌入式工程师而言内存消耗数据比任何性能描述都更具说服力。AceRoutine 的内存模型是其设计哲学的直接体现一切皆为静态分配零堆内存依赖。1.5.1 静态 RAMSRAM消耗组件AVR (8-bit)32-bit (ESP32/STM32)说明Coroutine实例16 字节28 字节固定开销与协程逻辑复杂度无关CoroutineScheduler2 字节4 字节单例与协程数量无关LogBinProfiler66 字节68 字节Profiling 功能的内存代价Channelint5 字节12 字节1 2 * sizeof(int)按平台对齐关键结论在 Nano 上增加一个协程仅增加16 字节 SRAM在 ESP32 上也仅增加28 字节。这对于仅有 2KB RAM 的 AVR 平台意味着可以轻松容纳数十个并发任务而不会引发任何内存危机。1.5.2 Flash程序存储器消耗Flash 消耗是另一个关键指标。AceRoutine 的设计确保了增量成本可控。以 Arduino Nano 为例基准程序1616 字节增加一个毫秒级协程188 字节增加两个毫秒级协程382 字节即第二个协程仅 194 字节这表明库具有良好的规模扩展性。增加协程带来的 Flash 增长并非线性爆炸而是渐进式增长这得益于宏系统对代码的高效复用。1.6 高级特性Profiling 与 Channel 的工程实践1.6.1 协程性能分析Profilingv1.5 引入的 Profiling 功能是 AceRoutine 从“可用”迈向“专业”的重要一步。它不再是一个黑盒而是提供了量化协程行为的工具。LogBinProfiler的核心思想是对数分桶Logarithmic Binning。它将runCoroutine()的执行时间划分为 32 个桶覆盖从 1 微秒到 4295 秒的广阔范围。每个桶的宽度是前一个桶的两倍1us, 2us, 4us, 8us...这种设计完美匹配了嵌入式系统中任务执行时间通常呈指数分布的特点既能精确捕捉微秒级的快速响应也能记录长时间的等待。// 在 setup() 中初始化 Profiler LogBinProfiler profiler1; blinkLed.setProfiler(profiler1); // 在 loop() 中使用带 Profiling 的运行方式 blinkLed.runCoroutineWithProfiling();LogBinTableRenderer和LogBinJsonRenderer则提供了两种直观的数据呈现方式方便开发者在串口监视器中实时观察协程的“心跳”是否健康。例如一个本应快速完成的 LED 控制协程如果其大部分执行时间落入了1ms甚至2ms的桶中就可能预示着其内部逻辑出现了意外的阻塞或计算密集型操作。1.6.2 通道Channel实验性但极具潜力的通信原语Channel是 AceRoutine 中唯一的实验性experimental特性其 API 和功能在未来版本中可能发生重大变更。然而其设计理念——为协程间提供一种同步、无缓冲、类型安全的消息传递机制——却直击嵌入式开发痛点。// 定义一个整数通道 Channelint dataChannel; // 在 Writer 协程中 COROUTINE(writer) { COROUTINE_LOOP() { int value readSensor(); COROUTINE_CHANNEL_WRITE(dataChannel, value); // 阻塞直到 Reader 读取 COROUTINE_DELAY(100); } } // 在 Reader 协程中 COROUTINE(reader) { COROUTINE_LOOP() { int value; COROUTINE_CHANNEL_READ(dataChannel, value); // 阻塞直到 Writer 写入 processValue(value); } }Channel的“无缓冲”特性是其精髓。它强制实现了生产者-消费者之间的严格同步消除了因缓冲区满/空而导致的竞态条件。一个典型的工程应用是构建一个“命令行解释器”CLI一个Reader协程从串口读取命令字符串一个Writer协程将解析后的命令结构体通过Channel发送给Executor协程。这种模式天然地将输入、解析、执行三个阶段解耦且保证了执行顺序的严格性。1.7 与主流方案的对比为何选择 AceRoutine在 Arduino 生态中多任务方案众多AceRoutine 的定位非常清晰。方案核心机制RAM 开销 (AVR)Flash 开销 (AVR)适用场景主要缺点AceRoutineStackless Coroutine (GCC computed goto)~16 字节/协程~180 字节/协程资源极度受限、大量简单任务、确定性实时要求无返回值、无局部变量持久化FreeRTOSPreemptive Multitasking (Hardware Timer)~200 字节/任务~5-10 KB复杂应用、需要优先级抢占、标准 POSIX APIRAM 消耗巨大、学习曲线陡峭Arduino-SchedulerStackful Coroutine (setjmp/longjmp)~128 字节/协程~1 KB/协程需要完整函数调用栈的复杂逻辑不支持 ESP 系列、栈空间固定且不可控ProtoThreads (Cosa)Stackless Coroutine (Duffs Device)~16 字节/协程~180 字节/协程同 AceRoutine但需迁移到 Cosa 框架非 Arduino 原生生态生态割裂AceRoutine 的独特价值在于它完美地填补了“裸机编程”与“全功能 RTOS”之间的空白。当你需要比millis()非阻塞轮询更优雅的代码结构又无法承受 FreeRTOS 的内存开销时AceRoutine 就是那个“刚刚好”的答案。它不是为了取代 FreeRTOS而是为了让你在 Nano、Pro Micro 这样的小板子上也能写出结构清晰、易于维护的“类操作系统”风格代码。2. 工程实践从入门到精通的代码范式2.1 Hello WorldCOROUTINE() 宏的典型应用以下是最经典的HelloCoroutine.ino示例它展示了 AceRoutine 的核心范式#include AceRoutine.h using namespace ace_routine; const int LED LED_BUILTIN; // 定义一个名为 blinkLed 的协程 COROUTINE(blinkLed) { COROUTINE_LOOP() { // 等价于 while(true) digitalWrite(LED, HIGH); COROUTINE_DELAY(100); // 挂起 100ms让其他协程运行 digitalWrite(LED, LOW); COROUTINE_DELAY(500); // 挂起 500ms } } // 定义另一个名为 printHelloWorld 的协程 COROUTINE(printHelloWorld) { COROUTINE_LOOP() { Serial.print(F(Hello, )); Serial.flush(); // 确保立即发送避免缓冲 COROUTINE_DELAY(1000); Serial.println(F(World)); COROUTINE_DELAY(4000); } } void setup() { delay(1000); Serial.begin(115200); while (!Serial); // 等待 Leonardo/Micro 的虚拟串口连接 pinMode(LED, OUTPUT); } void loop() { // 手动、顺序地运行所有协程 blinkLed.runCoroutine(); printHelloWorld.runCoroutine(); }此代码的精妙之处在于它用完全同步、线性的 C 代码实现了异步、并发的效果。blinkLed和printHelloWorld的代码逻辑互不干扰各自管理自己的状态mStateCOROUTINE_DELAY宏则像一个“魔法开关”在需要时暂停自己把 CPU 让给对方。整个loop()函数的执行时间极短几乎等于所有协程中runCoroutine()调用开销的总和而非它们延时时间的总和。2.2 进阶模式手动继承与状态管理当协程逻辑变得复杂需要维护多个状态变量时COROUTINE()宏的便捷性可能让位于手动继承的灵活性。class SensorReaderCoroutine : public Coroutine { private: uint32_t mLastReadTime; // 自定义状态变量 uint16_t mReadingCount; bool mIsCalibrating; public: SensorReaderCoroutine() : mLastReadTime(0), mReadingCount(0), mIsCalibrating(false) {} int runCoroutine() override { COROUTINE_LOOP() { // 状态机逻辑先校准再周期读取 if (mIsCalibrating) { calibrateSensor(); mIsCalibrating false; COROUTINE_DELAY(1000); } else { if (millis() - mLastReadTime 2000) { // 2秒周期 int value analogRead(A0); sendToChannel(value); mReadingCount; mLastReadTime millis(); } COROUTINE_DELAY(10); // 短暂让出避免忙等待 } } } private: void calibrateSensor() { /* ... */ } void sendToChannel(int value) { /* ... */ } }; // 在全局作用域创建实例 SensorReaderCoroutine sensorReader;手动继承的优势在于完全掌控构造函数可以在对象创建时进行一次性初始化如外设配置。灵活的状态管理可以定义任意数量和类型的私有成员变量它们的生命周期与协程对象完全一致。便于单元测试可以轻松地为SensorReaderCoroutine类编写 AUnit 测试用例。2.3 与 FreeRTOS 的协同混合调度策略在一些高端项目中开发者可能希望在 ESP32 这样的双核芯片上将 AceRoutine 与 FreeRTOS 结合使用以发挥各自所长。一个典型的模式是FreeRTOS 管理高优先级、硬实时任务如电机控制AceRoutine 管理低优先级、软实时任务如 UI 更新、日志记录。// 在 FreeRTOS 的一个高优先级任务中 void motorControlTask(void* pvParameters) { for(;;) { // 执行严格的 PID 控制循环 updateMotorPosition(); vTaskDelay(pdMS_TO_TICKS(1)); // 1ms 周期 } } // 在 FreeRTOS 的一个低优先级任务中运行 AceRoutine 的调度器 void aceRoutineTask(void* pvParameters) { CoroutineScheduler::setup(); // 初始化 AceRoutine 调度器 for(;;) { CoroutineScheduler::loop(); // 运行所有 AceRoutine 协程 vTaskDelay(pdMS_TO_TICKS(10)); // 每 10ms 调度一次降低 CPU 占用 } } // 在 setup() 中启动 FreeRTOS 任务 void setup() { xTaskCreate(motorControlTask, Motor, 2048, NULL, 3, NULL); xTaskCreate(aceRoutineTask, AceRoutine, 2048, NULL, 1, NULL); vTaskStartScheduler(); }在此模型中AceRoutine 的协程被“包裹”在一个 FreeRTOS 任务中它们共享该任务的栈空间但彼此之间仍是协作式调度。这既利用了 FreeRTOS 的强大调度能力和硬件中断支持又保留了 AceRoutine 的轻量与简洁。3. 限制与规避策略在约束中创造价值任何优秀的工具都有其适用边界。理解 AceRoutine 的限制并掌握相应的规避策略是将其成功应用于生产环境的关键。3.1 “无栈”带来的核心限制限制一协程无法返回值。原因runCoroutine()的返回值被用于内部状态机控制用户无法利用它传递业务数据。规避策略使用全局变量、类成员变量或通道Channel。例如一个读取温度的协程可以将结果写入一个static int gTemperature全局变量或更优地通过COROUTINE_CHANNEL_WRITE(tempChannel, tempValue)发送给处理协程。限制二协程无法保存局部栈变量。原因协程没有自己的栈COROUTINE_LOOP()内部的int i 0;在每次yield后都会丢失。规避策略将所有需要跨yield持久化的变量提升为类的私有成员变量如mCounter,mState或函数静态变量static int counter 0;。这是 AceRoutine 编程范式的基石。限制三协程必须静态分配。原因动态分配new会引入malloc/free在 AVR 上增加 600 字节 Flash并带来不可预测的堆碎片。规避策略在setup()或全局作用域中预先定义好所有协程实例。对于需要“动态创建”效果的场景可以预先定义一个协程数组并通过一个enabled标志位来逻辑上启用/禁用它们。3.2 实验性功能的风险管理Channel被明确标记为实验性其当前版本仅支持单生产者-单消费者SPSC模式。在生产环境中使用它必须进行充分的防御性编程// 错误假设 Channel 总是能成功读写 COROUTINE(writer) { COROUTINE_LOOP() { COROUTINE_CHANNEL_WRITE(chan, value); // 如果 reader 不存在此调用将永远阻塞 } } // 正确添加超时和错误检查需自行扩展 Channel 类 COROUTINE(writer) { COROUTINE_LOOP() { if (!chan.tryWrite(value, 1000)) { // 尝试写入超时 1000ms // 处理超时错误例如记录日志或重试 logError(Channel write timeout); } COROUTINE_DELAY(100); } }对于关键任务建议将Channel视为一个高级的、需要谨慎使用的“可选配件”而非核心基础设施。其成熟的应用案例如AceUtils库中的CommandLineInterface是学习其最佳实践的宝贵资源。4. 构建与部署从 IDE 到 CI/CD 的全流程4.1 Arduino IDE 集成AceRoutine 的安装极为简单打开 Arduino IDE进入工具 - 库管理器。搜索AceRoutine点击安装。IDE 会自动提示并安装其依赖库AceCommon。对于开发版可直接克隆 GitHub 仓库cd ~/Arduino/libraries git clone https://github.com/bxparks/AceRoutine.git git clone https://github.com/bxparks/AceCommon.git4.2 PlatformIO 配置在platformio.ini文件中添加[env:esp32dev] platform espressif32 board esp32dev framework arduino lib_deps bxparks/AceRoutine^1.5.1 bxparks/AceCommon^1.5.04.3 CI/CD 自动化测试利用 AceRoutine 自带的AUnit单元测试框架可以轻松集成到 CI 流程中。一个典型的 GitHub Actions 工作流如下name: Build and Test on: [push, pull_request] jobs: test: runs-on: ubuntu-latest steps: - uses: actions/checkoutv3 - name: Setup Arduino CLI uses: arduino/setup-arduino-cliv1 - name: Install Platforms run: | arduino-cli core install arduino:avr1.8.4 arduino-cli core install esp32:esp322.0.2 - name: Run Unit Tests run: | cd tests arduino-cli lib install --git-url https://github.com/bxparks/AceRoutine.git arduino-cli lib install --git-url https://github.com/bxparks/AceCommon.git # 使用 arduino-ci-test 或类似工具运行 AUnit 测试这套流程确保了每一次代码提交都能在多种硬件平台上得到验证极大地提升了库的可靠性。5. 总结一个嵌入式工程师的协程实践手册AceRoutine 不仅仅是一个 Arduino 库它是一套关于如何在资源地狱中构建优雅软件的哲学。它教会我们抽象的价值在于恰到好处不必追求大而全的 RTOS一个精巧的协程原语足以解决 80% 的嵌入式并发问题。性能是设计出来的不是优化出来的从computed goto的选择到static内存模型的坚持每一个决策都指向同一个目标——极致的效率。工程化就是管理权衡它坦诚地告诉你“不能做什么”无返回值、无栈变量并为你指明“应该怎么做”用成员变量、用 Channel。这种透明远比一个功能繁多却处处是坑的黑盒更有价值。对于一位正在为 Nano 上的 2KB RAM 而焦头烂额的工程师AceRoutine 提供的不是一个新玩具而是一把开启新世界大门的钥匙。它让你可以用COROUTINE_LOOP()和COROUTINE_DELAY()这样近乎自然语言的语法去构建一个脉搏清晰、节奏分明的嵌入式系统。这或许就是嵌入式开发最本真的魅力所在。

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

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

相关文章

SpringBoot-17-MyBatis动态SQL标签之常用标签

文章目录 1 代码1.1 实体User.java1.2 接口UserMapper.java1.3 映射UserMapper.xml1.3.1 标签if1.3.2 标签if和where1.3.3 标签choose和when和otherwise1.4 UserController.java2 常用动态SQL标签2.1 标签set2.1.1 UserMapper.java2.1.2 UserMapper.xml2.1.3 UserController.ja…

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…

网络编程(Modbus进阶)

思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …

华为云AI开发平台ModelArts

华为云ModelArts&#xff1a;重塑AI开发流程的“智能引擎”与“创新加速器”&#xff01; 在人工智能浪潮席卷全球的2025年&#xff0c;企业拥抱AI的意愿空前高涨&#xff0c;但技术门槛高、流程复杂、资源投入巨大的现实&#xff0c;却让许多创新构想止步于实验室。数据科学家…

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…