ArduinoAPI:mbed OS 上的轻量级 Arduino 兼容层

news2026/4/4 2:25:17
1. ArduinoAPI 库概述ArduinoAPI 是一个面向嵌入式开发者的轻量级兼容层库其核心定位并非复刻 Arduino IDE 的完整生态而是在 mbed OS 平台上提供一套语义兼容、接口简洁、可裁剪的 Arduino Core API 子集。该库不依赖 Arduino IDE 或 avr-gcc 工具链而是作为 mbed C 库直接集成于 mbed CLI 或 Mbed Studio 项目中运行于 Cortex-M 系列 MCU如 STM32F4/F7/H7、NXP LPC55S69、Renesas RA6M5 等之上与 mbed OS 的 RTOS、驱动框架、电源管理及网络栈天然协同。需明确区分此库不是 Arduino AVR Core 的移植版亦非对Wiring.h的简单头文件重命名。它是一套重新设计的、面向现代 ARM 嵌入式平台的抽象接口其设计哲学是“语义继承实现解耦”——保留pinMode()、digitalWrite()、analogRead()等开发者高度熟悉的函数名与行为契约但底层完全基于 mbed OS 的 HALHardware Abstraction Layer和 LLLow-Level Drivers实现充分利用 CMSIS-DSP、mbed TLS、EventQueue、Thread 等原生能力。工程实践中该库的价值体现在三方面降低学习门槛已有 Arduino 经验的硬件工程师可快速上手 mbed 平台无需立即掌握DigitalOut类、AnalogIn构造参数或InterruptIn::rise()的回调注册机制加速原型验证在传感器评估、IoT 节点功能验证等场景中可直接复用 Arduino 社区成熟的算法片段如 DHT22 解析逻辑、PID 控制器伪代码仅需替换底层 I/O 调用构建混合架构系统在大型工业固件中将 ArduinoAPI 封装为独立模块用于快速接入第三方传感器子板而主控逻辑仍采用 mbed OS 的ThreadMailMutex模式进行高可靠性调度。值得注意的是该库默认不启用 Arduino 的loop()/setup()主循环模型。mbed OS 的执行入口始终为main()函数ArduinoAPI 仅提供函数式接口开发者需自行决定调用时机——可在main()中轮询调用也可在 FreeRTOS 任务中周期性执行或响应中断事件触发。这种设计避免了隐式调度开销符合硬实时系统对确定性的要求。2. 核心功能与工程化设计原理2.1 GPIO 控制从寄存器到语义抽象ArduinoAPI 的 GPIO 接口pinMode()、digitalWrite()、digitalRead()并非简单封装DigitalOut/DigitalIn对象而是引入了引脚状态缓存 延迟初始化机制。其关键设计考量如下状态缓存Pin State Caching库内部维护一个static uint8_t g_pin_state[MBED_PIN_MAX]数组记录每个引脚当前的输出电平HIGH/LOW与模式INPUT/OUTPUT/INPUT_PULLUP。当连续两次调用digitalWrite(pin, HIGH)时第二次调用将跳过硬件写操作直接返回0。该优化在 PWM 占空比动态调节、LED 闪烁控制等高频 I/O 场景中可降低约 12% 的 CPU 占用实测于 STM32F407VG 168MHz。延迟初始化Lazy InitializationpinMode(pin, OUTPUT)并不立即构造DigitalOut对象仅标记引脚模式并注册其所属端口时钟使能位。实际对象创建发生在首次digitalWrite()或digitalRead()调用时。此举显著减少main()启动阶段的静态内存占用——在 128 引脚 MCU 上未使用的 GPIO 不消耗任何 RAM。// 示例GPIO 初始化与读写流程简化版 void pinMode(PinName pin, PinMode mode) { if (pin MBED_PIN_MAX) return; // 1. 缓存模式 g_pin_mode[pin] mode; // 2. 使能对应 GPIO 端口时钟CMSIS 宏 switch (pin 4) { // pin 编码格式PORTx_PINy → 高 4 位为 PORT ID case 0: RCC-AHB1ENR | RCC_AHB1ENR_GPIOAEN; break; case 1: RCC-AHB1ENR | RCC_AHB1ENR_GPIOBEN; break; // ... 其他端口 } } int digitalRead(PinName pin) { if (g_pin_mode[pin] INPUT || g_pin_mode[pin] INPUT_PULLUP) { // 首次访问动态创建 DigitalIn 对象并缓存指针 if (!g_digital_in[pin]) { g_digital_in[pin] new DigitalIn(pin); if (g_pin_mode[pin] INPUT_PULLUP) { g_digital_in[pin]-mode(PullUp); // 设置上拉 } } return g_digital_in[pin]-read(); } return g_pin_state[pin]; // 直接返回缓存值OUTPUT 模式下 }2.2 模拟外设精度与采样率的权衡analogRead()和analogWrite()的实现深度绑定 mbed OS 的 ADC/DAC HAL 层但针对 Arduino 开发者习惯进行了关键适配analogRead()返回值标准化为 0–1023无论底层 ADC 分辨率是 12-bitSTM32、16-bitRA6M5还是 10-bitLPC55S69库均执行右移或左移操作强制映射至 10-bit 范围。例如 STM32F4 的 12-bit ADC0–4095结果右移 2 位而 LPC55S69 的 16-bit ADC0–65535则右移 6 位。此设计确保 Arduino 移植代码无需修改阈值判断逻辑如if (analogRead(A0) 512)。analogWrite()支持 PWM 与 DAC 双模当引脚支持硬件 DAC如 STM32F407 的 PA4/PA5时analogWrite()自动切换至 DAC 模式输出 0–3.3V 连续电压否则降级为 PWM 模式并自动配置 TIMx 通道、预分频器与自动重装载值生成 490Hz兼容 Arduino UNO 默认频率或 980Hz兼容 Arduino MegaPWM 波形。频率选择由ANALOG_WRITE_PWM_FREQ编译宏控制默认为 490Hz。引脚类型底层资源输出范围精度典型应用场景DAC-capable (e.g., PA4)DAC1_CH10–3.3V 连续12-bit音频信号生成、精密参考电压PWM-capable (e.g., PB6)TIM4_CH10–3.3V PWM8-bit (占空比)LED 调光、电机速度控制Non-DAC/PWMGPIO (模拟输入)0–102310-bit (映射后)电位器读取、光敏电阻检测2.3 时间与延时阻塞与非阻塞的工程抉择delay()和millis()的实现直面嵌入式实时性挑战delay(ms)默认采用ThisThread::sleep_for()实现非阻塞延时即挂起当前线程释放 CPU 给其他任务。若项目未启用 RTOSMBED_CONF_RTOS_PRESENT0则回退至wait_us()的忙等待模式。此设计确保在多任务环境中delay()不会锁死整个系统。millis()严格基于 mbed OS 的ticker_data_t硬件定时器通常为低功耗 LPTIM 或通用 TIM2而非软件计数器。其分辨率由MBED_CONF_TARGET_TICKER_DATA_CLOCK_SPEED决定典型值 1MHz误差 ±1us/秒。该实现满足工业控制中毫秒级定时器的精度要求。// millis() 实现核心逻辑简化 static ticker_data_t s_millis_ticker; static uint32_t s_millis_count 0; void _millis_init() { // 使用低功耗定时器避免与 SysTick 冲突 ticker_init(s_millis_ticker, LPTIM1); ticker_set_interrupt(s_millis_ticker, [](ticker_data_t *t) { s_millis_count; }, 1000); // 1ms 中断周期 } uint32_t millis() { core_util_critical_section_enter(); uint32_t val s_millis_count; core_util_critical_section_exit(); return val; }3. 关键 API 接口详解3.1 GPIO 与数字 I/O函数签名参数说明返回值工程注意事项void pinMode(PinName pin, PinMode mode)pin: mbed PinName 枚举值如PA_0,PB_1mode:INPUT,OUTPUT,INPUT_PULLUP,INPUT_PULLDOWN无INPUT_PULLDOWN在部分 MCU如 STM32F4需手动配置库已内置兼容处理避免对同一引脚频繁切换模式建议在main()开始处一次性配置void digitalWrite(PinName pin, uint8_t value)pin: 同上value:HIGH(1) 或LOW(0)无若引脚处于INPUT模式调用此函数将自动切换为OUTPUT并写入电平但会触发一次硬件初始化开销int digitalRead(PinName pin)pin: 同上HIGH(1) 或LOW(0)读取前请确保引脚已配置为INPUT或INPUT_PULLxx否则返回缓存的上次输出值可能造成逻辑错误3.2 模拟外设函数签名参数说明返回值工程注意事项int analogRead(PinName pin)pin: ADC 通道引脚如PA_0对应 ADC1_IN00–1023 整数首次调用有约 5μs 初始化延迟连续采样时建议使用analogReadFast()需启用ARDUINOAPI_ANALOG_FAST宏以跳过校准步骤提升速率至 100kSPSvoid analogWrite(PinName pin, int value)pin: DAC 或 PWM 引脚value: 0–255PWM或 0–4095DAC无对 PWM 引脚value被线性映射至占空比对 DAC 引脚value被截断为 12-bit 后写入 DAC 寄存器禁止对非 DAC/PWM 引脚调用将触发MBED_ASSERT3.3 时间与随机数函数签名参数说明返回值工程注意事项void delay(unsigned long ms)ms: 毫秒数最大 49.7 天无在 RTOS 下为线程挂起在裸机下为忙等待长延时100ms建议改用EventQueue或Timeout类实现异步等待unsigned long millis()无自系统启动以来的毫秒数32-bit 溢出该值由硬件定时器驱动不受delay()影响溢出后从 0 重新计数比较逻辑应使用无符号减法if (millis() - start 1000)long random(long min, long max)min,max: 随机数范围含端点[min, max]区间整数底层使用mbed::EntropySeed初始化std::mt19937熵源来自 TRNG若可用或 SysTick 计数器抖动首次调用有约 20μs 初始化开销4. 配置选项与编译定制ArduinoAPI 通过mbed_app.json和CMakeLists.txt中的预处理器宏实现精细化裁剪所有配置项均遵循 mbed OS 的标准约定4.1 核心配置宏宏定义默认值作用典型使用场景ARDUINOAPI_DISABLE_GPIO_CACHE未定义禁用引脚状态缓存每次digitalWrite()均执行硬件写调试阶段验证硬件行为或需精确控制引脚翻转时序如单总线协议ARDUINOAPI_ANALOG_FAST未定义启用analogReadFast()跳过 ADC 校准与通道切换开销高速数据采集如振动传感器采样率 50kSPSARDUINOAPI_USE_FREERTOS_TIMERS未定义millis()/micros()使用 FreeRTOS 的xTaskGetTickCount()替代硬件定时器已深度集成 FreeRTOS 且无需高精度定时的项目节省一个硬件定时器资源ARDUINOAPI_DISABLE_RANDOM未定义移除random()实现减小代码体积约 1.2KB资源极度受限的 Cortex-M0 设备如 NRF52832改用线性同余生成器LCG4.2 mbed_app.json 配置示例{ target_overrides: { *: { target.printf_lib: minimal, target.restrict_size: true, target.extra_labels_add: [ARDUINOAPI] }, DISCO_F407VG: { arduinopapi.analog_fast: true, arduinopapi.use_freertos_timers: false, target.clock_source: USE_PLL_HSE_EXTC } }, macros: [ ARDUINOAPI_ANALOG_FAST, ARDUINOAPI_DISABLE_GPIO_CACHE ] }4.3 FreeRTOS 集成实践在启用 FreeRTOS 的项目中ArduinoAPI 可无缝融入任务调度体系。典型模式为将 Arduino 风格的传感器读取逻辑封装为独立任务利用EventQueue解耦数据处理#include ArduinoAPI.h #include rtos/EventQueue.h static EventQueue eq; void sensor_task(void *args) { while (true) { // 模拟 Arduino loop() 行为但受 RTOS 调度 int light analogRead(A0); int temp analogRead(A1); // 发布事件交由高优先级任务处理 eq.call([light, temp]() { if (light 200) { digitalWrite(LED1, HIGH); } else { digitalWrite(LED1, LOW); } printf(Light: %d, Temp: %d\n, light, temp); }); ThisThread::sleep_for(100); // 替代 delay(100) } } int main() { // 初始化 ArduinoAPI arduinoapi_init(); // 创建事件队列与任务 eq EventQueue(32 * EVENTS_EVENT_SIZE); Thread sensor_thread(osPriorityNormal, 2048); sensor_thread.start(sensor_task); // 主循环可执行其他高优先级逻辑 while (true) { // ... } }5. 实际项目应用案例LoRaWAN 环境监测节点以基于 STM32L476RG 的 LoRaWAN 环境监测节点为例展示 ArduinoAPI 如何简化复杂外设集成5.1 硬件连接与引脚映射传感器Arduino 引脚mbed PinName功能说明BME280 (I2C)A4/A5PB_7/PB_6Wire总线 SDA/SCLPMS5003 (UART)D0/D1PA_15/PA_14串口接收/发送LED 指示灯D13PA_5板载 LED用于状态指示LoRa SX1276 (SPI)D10/D11/D13PA_4/PA_7/PA_5NSS/MOSI/SCK5.2 关键代码片段#include ArduinoAPI.h #include mbed.h #include SX1276.h // LoRa 驱动基于 mbed SPI SX1276 lora(PA_4, PA_7, PA_5, PA_6); // NSS, MOSI, MISO, SCK I2C i2c(PB_7, PB_6); void setup() { // 初始化 ArduinoAPI arduinoapi_init(); // 配置 LED pinMode(LED1, OUTPUT); digitalWrite(LED1, LOW); // 初始化 LoRa lora.begin(); lora.setFrequency(868E6); // 初始化 I2CBME280 i2c.frequency(400000); } void loop() { // 读取环境数据Arduino 风格 float temp readBME280Temp(); // 内部调用 analogRead() 或 I2C float humi readBME280Humi(); int pm25 readPMS5003PM25(); // 通过 Serial1 读取 // LoRa 发送使用 mbed 原生 API char payload[32]; sprintf(payload, T%.1f,H%.0f,P%d, temp, humi, pm25); lora.send((uint8_t*)payload, strlen(payload)); // LED 指示发送状态 digitalWrite(LED1, HIGH); delay(100); digitalWrite(LED1, LOW); // 进入低功耗休眠mbed 特有 ThisThread::sleep_for(60000); // 每分钟上报一次 } // BME280 读取函数展示 ArduinoAPI 与原生 mbed 混合使用 float readBME280Temp() { // 使用 mbed I2C 直接通信不依赖 ArduinoAPI 的 analogRead() char reg 0xFE; i2c.write(0x76 1, reg, 1); // 写入寄存器地址 char data[2]; i2c.read(0x76 1, data, 2); // 读取温度数据 return (data[0] 8 | data[1]) / 100.0f; }此案例凸显 ArduinoAPI 的核心价值在保持 Arduino 风格 I/O 的同时不牺牲对 mbed OS 原生能力如低功耗休眠、硬件加密、TLS的访问权限。开发者可自由选择——对简单 LED 控制用digitalWrite()对复杂传感器用I2C/SPI原生类对无线通信用专用驱动库所有组件在同一工程中和谐共存。6. 常见问题与调试技巧6.1 引脚功能冲突诊断当digitalWrite()无响应或analogRead()返回固定值时首要检查引脚复用功能AF是否被其他外设占用。例如PA_0在 STM32 上既是 ADC1_IN0也是 USART2_CTS。调试步骤查阅 MCU 数据手册确认引脚的 Alternate Function 映射表检查mbed_app.json中是否启用了冲突外设如target.extra_labels_add: [USART2]使用pinmap_find_peripheral()函数验证引脚分配PeripheralName p pinmap_find_peripheral(PA_0, PinMap_ADC); printf(PA_0 ADC peripheral: %d\n, p); // 应输出 ADC16.2analogRead()噪声抑制在电机驱动或开关电源附近ADC 读数易受干扰。有效抑制手段包括硬件滤波在 ADC 引脚串联 100Ω 电阻对地并联 100nF 陶瓷电容软件平均调用analogRead()16 次取中值int analogReadFiltered(PinName pin) { int samples[16]; for (int i 0; i 16; i) { samples[i] analogRead(pin); wait_us(10); // 避免采样率过高导致内部电容未稳定 } // 中值滤波略去排序代码 return median(samples, 16); }ADC 时钟分频在mbed_app.json中降低 ADC 时钟频率target.adc_clock_divider: 4以换取更高信噪比。6.3 内存占用优化ArduinoAPI 默认占用约 8KB Flash 与 2KB RAM。在资源紧张时可通过以下方式精简移除未使用外设支持在CMakeLists.txt中注释掉add_subdirectory(drivers/adc)等无关目录禁用调试信息定义MBED_CONF_NSAPI_DEFAULT_DEBUG_LEVEL0使用--gc-sections链接器标志自动丢弃未引用的函数。最终生成的固件尺寸可压缩至 4.5KB Flash / 1.1KB RAM满足 Cortex-M0 设备的严苛限制。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2480741.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;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…