当前位置: 首页 > article >正文

别再只会用轮询了!GD32F103 USART中断与DMA传输实战对比(附代码)

GD32F103 USART通信三剑客轮询、中断与DMA的实战抉择在嵌入式开发中串口通信就像空气一样无处不在——调试信息输出、传感器数据采集、设备间通信都离不开它。但很多开发者停留在最基础的轮询方式就像只会用螺丝刀却面对一整套精密工具。本文将带你深入GD32F103的USART模块通过实测对比轮询、中断和DMA三种通信方式的性能差异帮你找到不同场景下的最优解。1. USART通信基础与三种模式概览USART通用同步异步收发器是嵌入式系统中使用最广泛的通信接口之一。在GD32F103系列中除了UART4外其他USART模块都支持三种数据传输模式轮询模式CPU不断查询状态寄存器等待数据传输完成中断模式数据传输完成后触发中断CPU在中断服务程序中处理数据DMA模式由DMA控制器自动搬运数据完全解放CPU这三种模式在代码复杂度、CPU占用率和传输效率上存在显著差异。下面这个表格直观展示了它们的核心特点特性轮询模式中断模式DMA模式CPU占用率100%忙等按需处理接近0%代码复杂度最简单中等较复杂延迟确定性低中高适用场景简单调试中低速常规应用高速大数据量传输最大理论波特率约500kbps约1Mbps可达4.5Mbps在实际项目中选择哪种模式需要综合考虑数据量、实时性要求和系统资源占用等因素。比如一个每秒钟只需要传输几十字节的温湿度传感器用轮询可能反而更简单直接而高速ADC采集系统则必须考虑DMA方案。2. 轮询模式简单但低效的实现轮询模式是大多数开发者最先接触的USART使用方式它的实现直白得像打开水龙头喝水——发送数据时等待发送缓冲区空接收数据时检查接收缓冲区非空标志。// 轮询方式发送字符串 void UART_SendString_Polling(uint32_t usart_periph, char* str) { while(*str) { usart_data_transmit(usart_periph, (uint8_t)*str); while(RESET usart_flag_get(usart_periph, USART_FLAG_TBE)); } } // 轮询方式接收数据 uint8_t UART_ReceiveByte_Polling(uint32_t usart_periph) { while(RESET usart_flag_get(usart_periph, USART_FLAG_RBNE)); return usart_data_receive(usart_periph); }这种模式的优点显而易见代码逻辑简单直观易于理解和调试不需要配置中断或DMA硬件依赖小适合在初始化阶段或简单任务中使用但它的缺点同样明显CPU资源浪费在等待传输完成期间CPU只能空转无法执行其他任务实时性差无法及时响应数据到达可能导致数据丢失效率低下在高波特率下CPU可能来不及处理连续到达的数据实测数据显示在115200波特率下轮询模式发送1KB数据需要约90ms期间CPU利用率始终维持在100%。而在处理接收数据时如果主程序有其他耗时操作很容易错过数据接收时机导致溢出错误。提示即使使用轮询模式也建议启用溢出错误中断至少能在数据丢失时及时发现问题。3. 中断模式平衡性能与复杂度的选择中断模式是轮询的升级版它通过硬件中断机制让CPU从不断查询中解放出来。当USART事件如数据接收完成、发送缓冲区空发生时硬件自动触发中断CPU暂停当前任务去处理数据传输。3.1 中断模式配置要点配置USART中断需要关注以下几个关键点NVIC设置配置中断优先级避免被其他高优先级中断阻塞中断事件选择根据需要启用发送完成、接收缓冲区非空等中断中断服务程序优化尽量保持ISR简短避免嵌套中断// 中断模式初始化示例 void USART_Interrupt_Init(uint32_t usart_periph) { // ...GPIO和USART基本配置与轮询模式相同 // 配置NVIC nvic_irq_enable(USART0_IRQn, 1, 0); // 抢占优先级1子优先级0 // 使能接收缓冲区非空中断 usart_interrupt_enable(usart_periph, USART_INT_RBNE); // 可选使能错误中断 usart_interrupt_enable(usart_periph, USART_INT_ERR); } // 中断服务程序 void USART0_IRQHandler(void) { if(RESET ! usart_interrupt_flag_get(USART0, USART_INT_FLAG_RBNE)) { uint8_t data usart_data_receive(USART0); // 将数据存入环形缓冲区 ring_buffer_put(rx_buf, data); } // 处理发送中断 if(RESET ! usart_interrupt_flag_get(USART0, USART_INT_FLAG_TBE)) { if(!ring_buffer_empty(tx_buf)) { uint8_t data ring_buffer_get(tx_buf); usart_data_transmit(USART0, data); } else { // 发送缓冲区空禁用发送中断避免持续触发 usart_interrupt_disable(USART0, USART_INT_TBE); } } }3.2 中断模式的性能特点中断模式相比轮询有了质的飞跃CPU利用率大幅降低实测在相同115200波特率下传输1KB数据CPU利用率降至约15%响应及时数据到达后立即触发中断几乎没有延迟系统吞吐量提升主程序可以并行处理其他任务但中断模式也有其局限性中断开销每次中断都有上下文保存/恢复的开销高频中断下可能成为瓶颈优先级管理不当的中断优先级设置可能导致关键任务被阻塞数据缓冲需要实现环形缓冲区等机制来处理突发数据在波特率超过1Mbps时中断模式开始显得力不从心。此时每个字节的传输时间不足10μs而典型的中断响应和处理时间可能达到1-2μsCPU大部分时间都在处理中断。4. DMA模式极致性能的终极方案DMA直接内存访问是USART通信的性能巅峰它通过专用硬件控制器在内存和外设间直接搬运数据完全不需要CPU介入。GD32F103的DMA控制器支持多达7个通道其中USART0_TX对应DMA0通道4USART0_RX对应DMA0通道5。4.1 DMA模式配置详解配置USART DMA传输需要同时设置DMA控制器和USART模块// DMA发送初始化 void USART_DMA_Tx_Init(uint32_t usart_periph, uint32_t dma_periph, uint32_t dma_channel) { // 启用DMA时钟 rcu_periph_clock_enable(RCU_DMA0); // 配置DMA通道 dma_parameter_struct dma_init_struct; dma_struct_para_init(dma_init_struct); dma_init_struct.direction DMA_MEMORY_TO_PERIPHERAL; dma_init_struct.memory_addr (uint32_t)send_buffer; dma_init_struct.memory_inc DMA_MEMORY_INCREASE_ENABLE; dma_init_struct.memory_width DMA_MEMORY_WIDTH_8BIT; dma_init_struct.number BUFFER_SIZE; dma_init_struct.periph_addr (uint32_t)USART_DATA(usart_periph); dma_init_struct.periph_inc DMA_PERIPH_INCREASE_DISABLE; dma_init_struct.periph_width DMA_PERIPHERAL_WIDTH_8BIT; dma_init_struct.priority DMA_PRIORITY_HIGH; dma_init(dma_periph, dma_channel, dma_init_struct); // 使能DMA通道 dma_channel_enable(dma_periph, dma_channel); // 配置USART使用DMA usart_dma_transmit_config(usart_periph, USART_DENT_ENABLE); } // DMA接收初始化类似主要区别在direction和periph/memory配置4.2 DMA模式性能实测DMA模式的性能优势令人印象深刻CPU占用趋近于零数据传输过程完全由硬件处理超高吞吐量实测在4.5Mbps波特率下仍能稳定传输支持大数据块单次传输可达65535字节下表对比了三种模式在115200波特率下传输1KB数据的性能表现指标轮询模式中断模式DMA模式传输时间(ms)89.288.788.5CPU占用率(%)100151最大中断次数010241代码复杂度★☆☆☆☆★★★☆☆★★★★☆虽然在小数据量时三种模式的传输时间相近但随着数据量增大DMA的优势会越来越明显。特别是在需要同时处理多个外设或复杂算法的系统中DMA解放的CPU资源可以显著提升整体性能。4.3 DMA使用中的坑与技巧内存对齐问题GD32的DMA对内存地址有对齐要求4字节传输时地址必须4字节对齐传输完成判断建议使用DMA中断而非USART TC标志来判断传输完成双缓冲技巧在高速连续传输时可以设置双缓冲区交替使用错误处理必须处理DMA传输错误中断特别是内存访问错误// DMA传输完成中断处理示例 void DMA0_Channel4_IRQHandler(void) { if(dma_interrupt_flag_get(DMA0, DMA_CH4, DMA_INT_FLAG_FTF)) { dma_interrupt_flag_clear(DMA0, DMA_CH4, DMA_INT_FLAG_FTF); // 处理传输完成逻辑 transfer_complete 1; } if(dma_interrupt_flag_get(DMA0, DMA_CH4, DMA_INT_FLAG_ERR)) { dma_interrupt_flag_clear(DMA0, DMA_CH4, DMA_INT_FLAG_ERR); // 处理传输错误 error_occurred 1; } }5. 实战选型指南与混合应用策略了解了三种模式的特性后如何在项目中做出正确选择以下是几个典型场景的建议5.1 单模式应用场景轮询模式适用初始化阶段的简单配置极低频率的调试信息输出资源极度受限的场合中断模式适用中低速常规通信1Mbps需要及时响应的控制指令不定长数据包处理DMA模式适用高速数据流1Mbps大块数据传输64字节低功耗要求的系统5.2 混合模式高级应用在实际项目中经常需要混合使用多种模式以获得最佳效果。以下是几种常见组合DMA发送中断接收适合命令响应型应用发送使用DMA提高效率接收用中断保证实时性中断控制DMA数据传输用中断处理控制指令大数据块传输切换到DMA模式双DMA通道全双工收发均使用独立DMA通道适合高速双向数据流// 混合模式示例DMA发送中断接收 void USART_Mixed_Init(void) { // 初始化USART和GPIO... // 配置DMA发送 USART_DMA_Tx_Init(USART0, DMA0, DMA_CH4); // 配置中断接收 nvic_irq_enable(USART0_IRQn, 1, 0); usart_interrupt_enable(USART0, USART_INT_RBNE); } // 发送函数改用DMA void UART_Send_DMA(uint8_t *data, uint16_t length) { // 等待上次传输完成 while(dma_flag_get(DMA0, DMA_CH4, DMA_FLAG_FTF) RESET); // 配置DMA传输 dma_channel_disable(DMA0, DMA_CH4); dma_memory_address_config(DMA0, DMA_CH4, (uint32_t)data); dma_transfer_number_config(DMA0, DMA_CH4, length); dma_channel_enable(DMA0, DMA_CH4); }5.3 GD32F103的特殊考量GD32F103的USART模块有几个需要特别注意的地方UART4不支持DMA只有USART0-3支持DMA功能时钟源差异USART0在APB2总线最高108MHz其他在APB1最高54MHz中断优先级DMA中断优先级通常应低于USART中断在资源受限的情况下可以巧妙利用USART的TXE发送缓冲区空和TC发送完成不同中断来实现高效传输。比如在中断模式下可以只在缓冲区空时装载数据而在最后使用TC中断来判断整个传输完成。6. 性能优化进阶技巧6.1 波特率极限挑战GD32F103的USART理论上支持最高4.5Mbps的波特率。要达到这个极限需要注意时钟配置确保系统时钟和APB总线时钟正确配置IO速度将USART引脚设置为最高速度模式GPIO_OSPEED_50MHZ信号完整性高频下需要关注PCB布线必要时添加终端电阻// 配置4.5Mbps波特率 usart_baudrate_set(USART0, 4500000U);6.2 低功耗优化在电池供电设备中USART通信的低功耗优化尤为重要空闲时关闭时钟不使用时禁用USART和DMA时钟DMA唤醒配置DMA传输完成中断唤醒MCU自动波特率检测某些型号支持自动波特率可节省配置时间6.3 错误处理与鲁棒性稳定的USART通信需要完善的错误处理机制启用所有错误中断溢出错误、噪声错误、帧错误等超时机制为DMA传输设置硬件或软件超时数据校验添加CRC或校验和验证数据完整性// 全面的错误中断配置 usart_interrupt_enable(USART0, USART_INT_ERR); usart_interrupt_enable(USART0, USART_INT_ORERR); usart_interrupt_enable(USART0, USART_INT_NERR); usart_interrupt_enable(USART0, USART_INT_FERR); usart_interrupt_enable(USART0, USART_INT_PERR);在GD32F103项目中使用USART时我曾遇到一个棘手问题在高波特率下偶尔会出现数据错位。最终发现是GPIO速度配置不足导致的将USART引脚设置为50MHz速度后问题解决。这个案例告诉我们高性能通信需要每个环节都优化到位。

相关文章:

别再只会用轮询了!GD32F103 USART中断与DMA传输实战对比(附代码)

GD32F103 USART通信三剑客:轮询、中断与DMA的实战抉择 在嵌入式开发中,串口通信就像空气一样无处不在——调试信息输出、传感器数据采集、设备间通信都离不开它。但很多开发者停留在最基础的轮询方式,就像只会用螺丝刀却面对一整套精密工具。…...

告别理论!用Minitab实战拆解CPK与PPK:从公式差异到实际生产报告解读

告别理论!用Minitab实战拆解CPK与PPK:从公式差异到实际生产报告解读 在工厂车间的日常质量管理中,CPK和PPK这两个指标常常让质量工程师们又爱又恨。爱的是它们能直观反映生产过程能力,恨的是当面对一份满是数字的报告时&#xff0…...

TensorFlowTTS多GPU训练终极指南:如何在大规模数据集上高效训练TTS模型

TensorFlowTTS多GPU训练终极指南:如何在大规模数据集上高效训练TTS模型 【免费下载链接】TensorFlowTTS :stuck_out_tongue_closed_eyes: TensorFlowTTS: Real-Time State-of-the-art Speech Synthesis for Tensorflow 2 (supported including English, French, Kor…...

猫抓浏览器插件:终极网页资源嗅探工具,轻松获取视频音频图片

猫抓浏览器插件:终极网页资源嗅探工具,轻松获取视频音频图片 【免费下载链接】cat-catch 猫抓 浏览器资源嗅探扩展 / cat-catch Browser Resource Sniffing Extension 项目地址: https://gitcode.com/GitHub_Trending/ca/cat-catch 你是否经常在网…...

别再手动找数据集了!用Python的openml库5分钟搞定机器学习数据加载(附实战代码)

用Python的openml库5分钟搞定机器学习数据加载(附实战代码) 还在为找数据集发愁?每次开始新项目都要花半天时间在Kaggle上筛选、下载、解压、清洗数据?今天介绍一个能让你彻底告别这些繁琐步骤的神器——openml库。这个Python库能…...

自动化测试工程师缺口扩大3倍:从业者的挑战、机遇与18个月黄金窗口期应对策略

行业结构性变革的十字路口当前,软件测试行业正处在一场深刻而剧烈的结构性变革之中。技术浪潮的迭代、业务模式的演进以及开发范式的迁移,共同推动着软件质量保障体系的全面重塑。一个不容忽视且日趋显著的信号是,市场对自动化测试工程师的需…...

TrafficMonitor插件大全:打造你的终极桌面监控中心

TrafficMonitor插件大全:打造你的终极桌面监控中心 【免费下载链接】TrafficMonitorPlugins 用于TrafficMonitor的插件 项目地址: https://gitcode.com/gh_mirrors/tr/TrafficMonitorPlugins 想在桌面上实时监控股票行情、硬件状态、天气信息却不想被复杂软件…...

离线环境或网络不佳?手把手教你本地部署Gazebo模型库,告别‘ground_plane缺失’错误

离线环境下的Gazebo模型库本地化部署实战指南 当你在实验室的封闭网络环境中启动Gazebo仿真时,看到机器人模型瞬间"坠入深渊"的场面,那种挫败感我深有体会。控制台不断刷新的"Unable to find uri[model://ground_plane]"错误提示&am…...

LiveEventBus安全与混淆配置:保护Android应用数据与代码的终极指南

LiveEventBus安全与混淆配置:保护Android应用数据与代码的终极指南 【免费下载链接】LiveEventBus :mailbox_with_mail:EventBus for Android,消息总线,基于LiveData,具有生命周期感知能力,支持Sticky,支持…...

CANoe测试模块怎么选?XML vs CAPL Test Module,我用700人投票结果告诉你

CANoe测试模块选择指南:XML与CAPL的深度对比与实战建议 在汽车电子测试领域,CANoe作为行业标杆工具,其测试模块的选择往往让新手工程师感到困惑。最近一项针对700名工程师的调研显示,70%的受访者倾向于使用XML Test Module&#x…...

AI Agent Harness自动化文档生成

AI Agent Harness自动化文档生成:从概念到实战的全面指南 关键词 AI Agent, 自动化文档生成, Harness框架, 大语言模型, 软件开发流程, DevOps, 技术文档 摘要 在当今快速发展的软件开发领域,文档编写往往被视为耗时且繁琐的工作。本文将深入探讨AI Agent Harness自动化文…...

别再硬编码了!用Avue的findObject方法动态更新表单选项(附完整代码示例)

动态表单进阶:Avue中findObject的实战应用与性能优化 在开发中后台管理系统时,表单的动态化需求几乎无处不在。想象这样一个场景:当用户选择不同租户时,角色、部门和岗位的选项需要实时变化。传统硬编码方式不仅难以维护&#xff…...

告别复制粘贴!手把手教你理解STM32F103C6T6点灯代码里的‘*(unsigned int *)0x4001100C’到底在干什么

从机器码到电子流动&#xff1a;解码STM32寄存器操作背后的硬件语言 当你第一次看到*(unsigned int *)0x4001100C & ~(1<<13);这样的代码时&#xff0c;是否感觉像在阅读外星文字&#xff1f;这串看似随机的数字和符号组合&#xff0c;实际上是连接软件世界与硬件物理…...

Mac用户如何通过12306ForMac实现高效抢票?四个核心功能详解

Mac用户如何通过12306ForMac实现高效抢票&#xff1f;四个核心功能详解 【免费下载链接】12306ForMac An unofficial 12306 Client for Mac 项目地址: https://gitcode.com/gh_mirrors/12/12306ForMac 还在为节假日抢不到火车票而烦恼吗&#xff1f;作为Mac用户&#xf…...

STM32密码锁项目复盘:我踩过的3个坑和优化思路(附完整工程)

STM32密码锁项目复盘&#xff1a;我踩过的3个坑和优化思路&#xff08;附完整工程&#xff09; 去年用STM32F103做了个密码锁&#xff0c;本以为按教程走就能轻松搞定&#xff0c;结果从按键扫描到Flash存储踩坑无数。现在把那些深夜调试的血泪教训和优化方案整理出来&#xff…...

Zydis:终极x86/x86-64反汇编器 - 如何快速入门轻量级代码分析

Zydis&#xff1a;终极x86/x86-64反汇编器 - 如何快速入门轻量级代码分析 【免费下载链接】zydis Fast and lightweight x86/x86-64 disassembler and code generation library 项目地址: https://gitcode.com/gh_mirrors/zy/zydis Zydis是一款快速轻量的x86/x86-64反汇…...

告别云服务器:用随身Wifi+Debian搭建PupBot,实现零月供的QQ机器人

随身WifiDebian打造零月供QQ机器人&#xff1a;私有化部署全指南 当大多数开发者习惯性地将服务部署在云端时&#xff0c;一个被忽视的事实是&#xff1a;我们正以数据安全和系统控制权为代价换取所谓的便利性。本文将带你探索一种颠覆性的解决方案——利用随身Wifi设备配合Deb…...

为什么Windows用户需要Coolapk-UWP桌面客户端?

为什么Windows用户需要Coolapk-UWP桌面客户端&#xff1f; 【免费下载链接】Coolapk-UWP 一个基于 UWP 平台的第三方酷安客户端 项目地址: https://gitcode.com/gh_mirrors/co/Coolapk-UWP 你是否厌倦了在小屏幕上浏览酷安社区&#xff1f;是否希望在电脑大屏幕上也能享…...

如何使用rsync实现实时文件同步:inotify配置与自动备份完整指南

如何使用rsync实现实时文件同步&#xff1a;inotify配置与自动备份完整指南 【免费下载链接】rsync An open source utility that provides fast incremental file transfer. It also has useful features for backup and restore operations among many other use cases. 项…...

别再只盯着SBC了!手把手教你为安卓/Windows电脑开启AptX HD和LDAC蓝牙高清音频

解锁高清蓝牙音质&#xff1a;安卓与Windows系统开启AptX HD/LDAC全攻略 当你在通勤路上用蓝牙耳机听歌时&#xff0c;是否总觉得音质单薄、细节缺失&#xff1f;这很可能是因为你的设备默认使用了基础的SBC编码。实际上&#xff0c;现代蓝牙技术已经支持AptX HD和LDAC等高解析…...

别再手动改仿真值了!用LabVIEW 2020 + mbslave实现Modbus TCP数据自动读写与监控

LabVIEW 2020与Modbus TCP自动化监控实战指南 在工业自动化测试领域&#xff0c;手动修改仿真参数的时代已经过去。想象一下这样的场景&#xff1a;凌晨三点的生产线突然出现异常&#xff0c;而你的系统能够自动捕捉数据变化、触发警报并记录完整的过程数据——这正是现代自动…...

InstantSearch 高级技巧:10个提升搜索性能的实用方法

InstantSearch 高级技巧&#xff1a;10个提升搜索性能的实用方法 【免费下载链接】instantsearch ⚡️ Libraries for building performant and instant search and recommend experiences with Algolia. Compatible with JavaScript, TypeScript, React and Vue. 项目地址: …...

别再只看信号格了!5G网速上不去?手把手教你用RSRP、SNR、BLER看懂手机里的真实信道质量

手机信号满格却网速慢&#xff1f;5G时代必懂的RSRP、SNR、BLER诊断指南 你是否遇到过这样的情况&#xff1a;手机信号显示满格&#xff0c;但刷视频却频繁缓冲&#xff0c;下载文件速度慢如蜗牛&#xff1f;这种"信号假象"在5G时代尤为常见。本文将带你揭开手机信号…...

SpringCloud Alibaba微服务排错实战:用SkyWalking揪出那个拖慢接口的“慢SQL”

SpringCloud Alibaba微服务排错实战&#xff1a;用SkyWalking揪出那个拖慢接口的"慢SQL" 问题现象&#xff1a;接口响应时间突然飙升 那天下午3点17分&#xff0c;我正喝着咖啡准备处理下一个需求&#xff0c;突然收到监控系统告警&#xff1a;订单查询接口的P99响应…...

urllib3 性能优化终极指南:7个提升HTTP请求速度的实用技巧

urllib3 性能优化终极指南&#xff1a;7个提升HTTP请求速度的实用技巧 【免费下载链接】urllib3 urllib3 is a user-friendly HTTP client library for Python 项目地址: https://gitcode.com/gh_mirrors/ur/urllib3 urllib3 是 Python 生态中最受欢迎的 HTTP 客户端库之…...

多人协同报价单系统|跨行业通用、支持图片上传与PDF导出

温馨提示&#xff1a;文末有联系方式多人协同报价单功能全面升级 本报价单系统专为团队协作设计&#xff0c;支持局域网环境下的多人同时在线操作&#xff0c;无需复杂部署&#xff0c;即装即用&#xff0c;大幅提升与商务部门协同效率。跨行业通用型报价单模板 无论您身处制造…...

如何在5分钟内快速安装和运行StarSpace:终极初学者指南

如何在5分钟内快速安装和运行StarSpace&#xff1a;终极初学者指南 【免费下载链接】StarSpace Learning embeddings for classification, retrieval and ranking. 项目地址: https://gitcode.com/gh_mirrors/st/StarSpace StarSpace是一款强大的学习嵌入工具&#xff0…...

曦智科技招股:拟募资25亿港元 要做全球AI硅光芯片第一股

雷递网 乐天 4月20日光电混合算力提供商——上海曦智科技股份有限公司&#xff08;简称“曦智科技”&#xff09;今日正式启动H股全球发售计划&#xff0c;将于4月23日截止认购&#xff0c;并预计于4月28日正式以股票代码“01879.HK”挂牌上市&#xff0c;全力冲刺全球资本市场…...

算法视角的职场破局:如何重塑 LinkedIn 画像,捕获全球跨国企业 HR 的搜索雷达

在留学生求职的日常中&#xff0c;我们经常会听到这样的困惑&#xff1a;精心打磨了单页简历&#xff0c;每天坚持在各大公司的招聘官网上投递&#xff0c;结果往往是石沉大海&#xff1b;虽然早早注册了 LinkedIn&#xff08;领英&#xff09;账号&#xff0c;但除了偶尔添加几…...

JD-GUI:Java字节码反编译的终极图形化解决方案

JD-GUI&#xff1a;Java字节码反编译的终极图形化解决方案 【免费下载链接】jd-gui A standalone Java Decompiler GUI 项目地址: https://gitcode.com/gh_mirrors/jd/jd-gui 对于Java开发者来说&#xff0c;面对只有编译后的.class文件却需要理解其内部逻辑的情况并不少…...