基于CircuitPython与RP2350的嵌入式多声道音频系统设计与实践

news2026/5/16 2:27:09
1. 项目概述用CircuitPython打造你的专属交互式音频系统如果你玩过树莓派Pico或者Adafruit的Feather系列开发板可能会觉得在微控制器上处理音频是件挺麻烦的事——要么得用专门的解码芯片要么代码复杂得让人头疼。但最近我在一个互动艺术装置项目里需要实现一个能根据按钮触发不同音效甚至能同时播放多个背景声音的系统这让我重新审视了嵌入式音频的方案。传统的方案比如Adafruit经典的Audio FX系列板卡基于VS1000芯片确实简单易用插上SD卡、接上按钮就能响。但它有个硬伤功能固定只能单声道播放而且芯片型号也有些年头了。当项目需要更灵活的触发逻辑或者想实现声音的叠加比如按下按钮时背景音乐不中断时就显得力不从心了。这正是我转向CircuitPython Audio FX这个方案的原因。它本质上是一个用CircuitPython写的软件库跑在像Raspberry Pi Pico 2RP2350核心或Adafruit Feather RP2350这样的现代微控制器上。它的巧妙之处在于你完全不需要写复杂的代码来控制音频播放逻辑。你只需要按照特定的规则给音频文件命名然后复制到开发板的存储里系统就能自动识别并根据连接按钮的按下/释放事件执行播放、循环、随机播放等操作。更棒的是得益于RP2350更强的处理能力它甚至能实现多声道Polyphonic播放也就是同时播放多个音频文件这对于创建丰富的声景体验至关重要。这套系统非常适合用来快速原型开发各种需要声音反馈的交互项目比如互动展览装置、有声故事书、智能玩具、或者自定义的音效板。即使你没有深厚的嵌入式开发背景只要会用电脑复制文件、接几根线就能让想法“发声”。接下来我就带你从硬件选型到软件配置完整地走一遍构建流程并分享一些我从实际项目中总结出来的、文档里不会写的细节和避坑指南。2. 核心硬件选型与电路连接解析工欲善其事必先利其器。一个稳定可靠的硬件基础是项目成功的前提。这一部分我们不仅要知道“用什么”更要明白“为什么用这个”以及如何正确地连接它们。2.1 微控制器为什么是RP2350项目的核心是微控制器。原文提到了两款板子Raspberry Pi Pico 2和Adafruit Feather RP2350。它们都采用了瑞萨Raspberry Pi Foundation最新的RP2350双核微控制器。选择RP2350而不是更常见的RP2040或其他型号主要基于三个关键考量更强的计算能力与内存多声道音频解码是计算密集型任务。RP2350相比RP2040有更高的主频和更大的RAM这对于同时运行多个MP3软件解码器audiomp3.MP3Decoder对象至关重要。每个解码器实例都需要占用不少内存来存放解码状态和缓冲区。充足的Flash存储Pico 2板载的Flash通常被划分为两部分一部分存放CircuitPython固件和代码另一部分作为CIRCUITPY虚拟U盘。这个U盘空间就是你存放音频文件的地方。RP2350方案能提供约3MB的可用空间这对于存储压缩后的MP3音效来说已经相当充裕。完善的CircuitPython生态支持Adafruit对RP2350的CircuitPython支持非常到位audiobusio、audiomp3、audiomixer等关键音频库都已适配提供了高级别的硬件抽象让我们免于直接操作寄存器之苦。实操心得如果你手头只有RP2040的板子比如Pico W理论上也能运行单声道Monophonic版本的代码因为计算负载小很多。但如果你想体验多声道同时播放的乐趣或者音频文件稍长、码率稍高RP2350会是更稳妥、体验更好的选择。投资一块Pico 2在音频项目上绝对物有所值。2.2 音频输出I2S DAC的必要性微控制器本身通常没有直接的模拟音频输出能力。我们需要一个数模转换器DAC将数字音频信号转换成模拟信号才能驱动耳机或扬声器。这里我们采用I2SInter-IC Sound协议它是一种专门为数字音频传输设计的串行总线标准。我选用的是Adafruit I2S Stereo Decoder - UDA1334A Breakout。原因如下集成度高这块板子集成了UDA1334A DAC芯片、必要的滤波电路和3.5mm耳机接口开箱即用。音质有保障UDA1334A是一款性能不错的立体声DAC支持16/24/32位深度和多种采样率对于本项目绰绰有余。电路简单与微控制器仅需3根数据线BCLK, WSEL, DAT和共地连接无需复杂的模拟电路设计。当然你也可以使用其他兼容I2S的DAC模块或者甚至使用带I2S输入的功放模块如MAX98357直接驱动扬声器。核心是找到支持I2S输入、且与3.3V逻辑电平兼容的设备。2.3. 电路连接详解与避坑指南连接原理很简单但细节决定成败。下图展示了核心的连接关系Raspberry Pi Pico 2 / Feather RP2350 ┌─────────────────────────────────────┐ │ │ │ GP16 ────────────── BCLK │ │ GP17 ────────────── WSEL (LRCLK)│ │ GP18 ────────────── DATA │ │ GND ────────────── GND │ │ │ │ GP0..GP15 ───┐ │ │ │ [按钮/开关] │ │ └── 一端接GPx │ │ 另一端接GND │ └─────────────────────────────────────┘ │ ▼ Adafruit UDA1334A I2S DAC ┌─────────────────────────┐ │ │ │ BCLK ────────────────┘ │ WSEL ────────────────┘ │ DATA ────────────────┘ │ GND ────────────────┘ │ │ │ 3.3V/5V ──── VIN │ │ GND ──── GND │ │ │ │ 音频输出 ─── 3.5mm接口 │ └─────────────────────────┘连接步骤与关键细节I2S连接3根线 GNDBCLK (Bit Clock)位时钟连接至Pico的GP16。这根线告诉DAC每个数据位的时间边界。WSEL (Word Select / LRCLK)字选择或左右声道时钟连接至GP17。它告诉DAC当前传输的是左声道数据还是右声道数据。DATA串行音频数据连接至GP18。实际的音频采样数据在这根线上传输。GND务必用一根导线将微控制器的GND引脚与DAC模块的GND连接起来为信号提供共同的参考地这是消除噪音的基础。电源连接给DAC模块供电。UDA1334A板子有一个VIN引脚可以接受3.3V-5V的输入。你可以从Pico的3V3(OUT)引脚取电最大输出电流需查看板子规格或者使用外部电源如USB供电单独为DAC供电。如果使用外部电源必须确保外部电源的GND与Pico的GND相连这是很多噪音问题的根源。触发按钮连接最多16个使用常开型按钮或拨动开关。一端连接到Pico的GP0至GP15中的任意一个GPIO引脚。另一端直接连接到Pico的GND引脚。代码中通过keypad.Keys配置为value_when_pressedFalse这意味着当按钮按下引脚与GND短路时系统会检测到一个“低电平”或“False”事件从而触发播放。重要注意事项上拉电阻Pico的GPIO内部通常有可配置的上拉电阻。在我们的代码配置下value_when_pressedFalsekeypad库会默认启用内部上拉电阻。这意味着当按钮未按下时GPIO引脚被内部电阻拉高到3.3V逻辑1按下时被拉低到GND逻辑0。因此你不需要在外部额外添加电阻。防抖keypad库已经内置了软件防抖处理可以有效避免因按钮触点机械抖动导致的多次误触发。所以我们也无需额外添加硬件电容进行防抖。引脚冲突确保你用于I2S和按钮的GPIO没有其他复用功能如用于连接其他传感器。GP16、GP17、GP18是Pico上常用的I2S引脚通常很安全。3. 软件环境部署与项目文件解析硬件搭好了接下来就是让系统“活”起来。这一步涉及到固件、库和项目文件的部署是项目能否运行的关键。3.1 搭建CircuitPython开发环境安装CircuitPython固件访问CircuitPython官网找到对应你的开发板如Raspberry Pi Pico 2或Feather RP2350的最新稳定版.uf2固件文件。按住开发板上的BOOT或BOOTSEL按钮同时通过USB线将开发板连接到电脑。然后松开按钮。此时电脑会识别出一个名为RPI-RP2对于Pico或类似的可移动磁盘。将下载好的.uf2固件文件拖入这个磁盘。开发板会自动重启之后磁盘名称会变为CIRCUITPY。这表明CircuitPython系统已经成功刷入。获取项目代码与库文件根据原文指引从Adafruit学习系统页面找到CircuitPython Audio FX项目点击“Download Project Bundle”。这会下载一个包含所有必需文件的ZIP压缩包。解压后你会看到针对不同CircuitPython版本如9.x的文件夹以及lib库文件文件夹。部署文件到CIRCUITPY磁盘打开CIRCUITPY磁盘。将对应你CircuitPython版本的文件夹内的所有内容主要是code.py复制到CIRCUITPY盘的根目录。将lib文件夹内的所有.mpy库文件如adafruit_bus_device,adafruit_ticks,neopixel.mpy等但最重要的是audiobusio,audiocore,audiomp3,audiomixer和keypad复制到CIRCUITPY盘的lib文件夹内。如果lib文件夹不存在就新建一个。此时你的CIRCUITPY磁盘根目录下至少应有code.py和lib文件夹。避坑技巧库文件版本必须与CircuitPython固件版本匹配。使用项目包中提供的库是最保险的。如果你从别处单独下载库不匹配的版本可能会导致ImportError或运行时错误。一个常见的错误是忘记复制audiomixer.mpy这个库是多声道混音的核心没有它多声道功能无法工作。3.2 代码框架深度解析项目提供了两个版本的code.py多声道Polyphonic和单声道Monophonic。我们以功能更强大的多声道版本为例深入其核心架构。核心设计思想这是一个事件驱动、基于文件命名约定的系统。程序启动时会扫描CIRCUITPY根目录下所有符合Txx[类型].mp3/.wav命名规则的文件并为每个检测到的文件在对应的GPIO引脚上创建一个“触发器Trigger”对象。主循环不断检测按钮事件并调用相应触发器的on_press()或on_release()方法。让我们拆解几个关键代码段1. 硬件与资源配置pads [board.GP0, board.GP1, ..., board.GP15] # 按钮引脚定义 max_simultaneous_voices 2 # 最大同时发声数复音数 audiodev audiobusio.I2SOut(bit_clockboard.GP16, ...) # I2S音频设备pads列表定义了哪些GPIO用于按钮。顺序代表优先级当需要停止一个正在播放的声音来给新声音让路时复音数已满系统会停止列表中索引号最大即位置靠后的触发器的声音。所以如果你把最重要的背景音乐触发器放在GP0把一次性的音效放在GP15那么背景音乐被意外中断的概率就最小。max_simultaneous_voices是软件层面的复音数上限。但注意MP3解码器还有另一个硬件限制decoders [audiomp3.MP3Decoder(...) for _ in range(min(4, max_simultaneous_voices))]。这里创建了最多4个MP3解码器对象。这意味着即使你设置max_simultaneous_voices8你最多也只能同时播放4个MP3文件因为RAM限制。WAV文件解码负担轻不受此限但WAV文件体积大。2. 触发器Trigger类体系这是整个系统的灵魂。所有触发器都继承自TriggerBase。其工作流程是__init__: 根据文件名前缀如T00扫描匹配的音频文件。on_press(): 按钮按下时调用。通常会调用self.play()。play(): 核心播放方法。它会先尝试force_off()停止自己可能正在播放的声音然后通过ensure_available_voice()和ensure_available_decoder()获取可用的“声音槽”和“解码器槽”最后启动播放。force_off(): 停止播放并释放占用的“声音槽”和“解码器槽”使其可供其他触发器使用。系统预定义了五种触发器类型其行为完全由文件名中的“词干stem”决定BasicTrigger(词干:): 例如T00.mp3。按下按钮播放一次文件。HoldLoopingTrigger(词干:HOLDL): 例如T01HOLDL.mp3。按住按钮时循环播放松开即停。LatchingLoopTrigger(词干:LATCH): 例如T02LATCH.mp3。按一下开始循环播放再按一下停止。像是一个播放/暂停开关。PlayNextTrigger(词干:NEXT0~NEXT9): 例如T03NEXT0.mp3,T03NEXT1.mp3,T03NEXT2.mp3。每次按下按钮会按顺序播放NEXT0,NEXT1,NEXT2然后回到NEXT0实现音效序列。PlayRandomTrigger(词干:RAND0~RAND9): 例如T04RAND0.mp3,T04RAND1.mp3,T04RAND2.mp3。每次按下按钮随机从RAND0~RAND2中选一个播放。3. 资源管理与优先级抢占这是实现多声道的核心机制。系统维护着两个队列available_voices可用声音槽和available_decoders可用MP3解码器。当一个新的播放请求到来时ensure_available_voice()和ensure_available_decoder()会首先检查队列里是否有空闲资源。如果没有它们会按照reversed_triggers即反向的触发器列表从高索引号到低索引号的顺序强制停止(force_off)一个正在播放的触发器以释放其资源。这意味着低编号引脚高优先级的触发器可以抢占高编号引脚低优先级的触发器的播放资源。你需要根据这个逻辑来规划你的音效重要性。4. 音频文件制备从原理到实践音频文件是系统的“弹药”。制备不当轻则播放异常重则系统崩溃。这部分我们深入聊聊格式、参数和命名规则背后的门道。4.1 格式、编码与参数详解系统支持MP3和WAV格式。选择哪种取决于你的存储空间和音质要求。WAV文件无损格式音质完美但体积巨大。一个16kHz、16位、单声道的WAV文件每秒约占用32KB空间16k samples/s * 2 bytes/sample。3MB的存储空间只能存约1.5分钟的音频。仅推荐用于极短的提示音或作为转换MP3的源文件。MP3文件有损压缩格式能在保证可接受音质的前提下大幅减小体积。这是本项目的主流选择。多声道版本的关键限制为了能够将多个音频流混合成一个输出所有音频文件必须具有完全相同的音频规格。这包括采样率Sample Rate如16000 Hz, 22050 Hz, 44100 Hz。所有文件必须统一。声道数Channel Count单声道Mono1 channel或立体声Stereo2 channels。所有文件必须统一。强烈建议使用单声道因为立体声文件数据量翻倍对解码和混音压力更大且对于大多数音效和背景声单声道足够。位深度Bits Per Sample必须是16位有符号整数16-bit signed。这是audiomixer.Mixer的要求。单声道版本则无此限制可以混合不同采样率和声道数的文件因为它一次只播放一个文件不涉及混音。4.2 使用Audacity进行音频处理实战步骤这里以最常用的免费开源软件Audacity为例演示如何将一个普通的音频文件处理成项目可用的格式。导入源文件打开Audacity将你的音效或音乐文件拖入。标准化声道如果源文件是立体声点击轨道左侧的倒三角选择“拆分立体声轨道”。通常音效放在单声道即可。你可以删除其中一个声道或者选中两个声道点击菜单轨道(T)-混音-将立体声轨道渲染为单声道。这样会合并成一个单声道轨道。统一采样率在左下角可以看到当前项目的采样率如44100 Hz。点击它将其改为目标采样率例如16000 Hz。这是一个在音质和文件大小/解码压力之间很好的平衡点。更改后Audacity会进行重采样。修剪与调整裁剪掉不需要的静音部分调整音量效果 - 放大/压缩器确保峰值不要超过0dB避免 clipping破音。导出为MP3点击文件-导出-导出为MP3。在导出设置中关键步骤来了比特率模式选择“恒定比特率”。质量/比特率这是文件大小和音质的杠杆。对于语音或简单音效32 kbps足够清晰且体积小。对于音乐可以考虑64 kbps或96 kbps。记住比特率越低同时播放多个文件时系统压力越小能存储的音频总时长也越长。声道模式选择“单声道”。即使你上一步处理成了单声道这里也要确保导出为单声道MP3。点击保存填写ID3标签可选完成导出。核心经验“低比特率 低采样率 单声道”是嵌入式音频的黄金法则。一个16kHz单声道、32kbps的MP3文件每分钟仅占约240KB。3MB的存储空间可以放下超过12分钟的音频这对于绝大多数交互项目来说已经非常充裕。盲目使用“CD音质”44.1kHz立体声128kbps会迅速耗尽存储并可能使多声道播放变得卡顿。4.3 文件命名规则与实战案例命名规则是控制播放行为的“魔法咒语”。格式为T[两位触发编号][触发类型词干].[扩展名]。触发编号00到15对应代码中pads列表的索引。T00对应GP0T01对应GP1以此类推。触发类型词干决定播放逻辑如基本、HOLDL、LATCH、NEXT0、RAND0等。扩展名.mp3或.wav。案例配置 假设我们有一个小型互动故事机设计了6个按钮GP0 (T00): 背景环境音循环。文件T00LATCH.mp3GP1 (T01): 主角打招呼语音随机三种。文件T01RAND0.mp3,T01RAND1.mp3,T01RAND2.mp3GP2 (T02): 风声特效按住播放。文件T02HOLDL.mp3GP3 (T03): 任务提示语音顺序播放三步。文件T03NEXT0.mp3,T03NEXT1.mp3,T03NEXT2.mp3GP4 (T04): 正确反馈音。文件T04.mp3GP5 (T05): 错误反馈音。文件T05.mp3将以上8个MP3文件注意T01和T03各有多个文件全部复制到CIRCUITPY磁盘根目录。系统启动时会自动扫描并建立映射。按下GP1的按钮就会随机播放一个打招呼语音按住GP2的按钮风声持续响起点击一下GP0背景音乐开始循环播放再点击一下停止。5. 高级配置、调试与性能优化系统搭建起来并能播放声音只是第一步。要让它在实际项目中稳定、可靠、表现符合预期还需要进行一些调试和优化。5.1 代码自定义与扩展原版code.py已经非常强大但你完全可以按需修改1. 修改引脚定义和复音数# 如果你只用了4个按钮可以这样定义节省一点资源 pads [board.GP0, board.GP1, board.GP2, board.GP3] # 增加复音数到4允许更多声音叠加注意MP3解码器上限 max_simultaneous_voices 42. 创建自定义触发器类型假设你想要一个“按下播放但播放完当前文件才响应下一次按下”的触发器防止音效重叠你可以继承TriggerBase创建新类class PlayOnceWaitTrigger(TriggerBase): 按下播放如果正在播放则忽略本次按下 stems [WAIT] def on_press(self): if not self.playing: # 只有不在播放时才触发 self.play(self.filenames[0]) def on_release(self): pass # 别忘了把它添加到 trigger_classes 列表中 trigger_classes [ BasicTrigger, HoldLoopingTrigger, LatchingLoopTrigger, PlayNextTrigger, PlayRandomTrigger, PlayOnceWaitTrigger, # 添加自定义触发器 ]然后一个命名为T06WAIT.mp3的文件就会使用这个新逻辑。5.2 串口调试与问题排查当系统没有按预期工作时串口调试REPL是你的最佳伙伴。用串口工具如Mu编辑器、Thonny、或者screen/putty连接到开发板的串口通常波特率115200。程序启动时会打印大量信息扫描到的触发器列表。可用的解码器和声音槽ID。每次按钮事件按下/释放的详细信息。资源分配和释放的日志如force off,return voice,return decoder。常见问题速查表现象可能原因排查步骤完全没声音1. I2S接线错误或接触不良。2. DAC未供电或供电错误。3. 音频文件格式/参数不符。1. 检查BCLK/WSEL/DATA/GND四根线。2. 用万用表测量DAC的VIN和GND间电压。3. 查看REPL启动日志确认是否识别到音频文件。检查文件规格是否一致多声道版本。播放卡顿、爆音1. 复音数(max_simultaneous_voices)设置过高。2. MP3比特率或采样率过高。3. 音频文件本身有问题。1. 在REPL中观察播放时是否频繁触发force_off。尝试降低max_simultaneous_voices到2或1。2. 将所有音频转换为16kHz单声道、32kbps MP3再试。3. 在电脑上用播放器检查音频文件是否正常。某个按钮无反应1. 该引脚对应的音频文件缺失或命名错误。2. 按钮接线错误或损坏。3. 该引脚被其他功能占用。1. 检查CIRCUITPY根目录下是否有对应Txx...的文件。2. 用万用表通断档检查按钮按下时对应GPIO引脚是否与GND短路。3. 检查code.py中pads列表是否包含了该引脚。播放声音音调不对过快/过慢音频文件采样率与代码中Mixer初始化时检测到的采样率不匹配。确保所有音频文件的采样率完全相同。用Audacity等工具统一转换。只能播放一个声音第二个声音会打断第一个max_simultaneous_voices被设置为1。检查并修改code.py中的max_simultaneous_voices变量确保其大于1。5.3 性能优化与边界探索RP2350的性能有其边界通过优化可以挖掘最大潜力超频CircuitPython支持超频。在code.py开头或boot.py中添加import microcontroller; microcontroller.cpu.frequency 250_000_000将CPU频率提升到250MHz。这可以显著提升多声道解码能力可能让你能同时稳定播放3-4个中等码率的MP3。注意超频可能增加功耗和发热且不一定在所有板子上稳定。内存管理代码中硬编码了最多4个MP3解码器(decoders列表)。如果你确信你的应用场景中永远不会同时播放超过2个MP3可以将这个数字减到2以节省一些RAM。但通常不建议修改除非你遇到内存不足的错误。文件系统优化CIRCUITPY文件系统通常是FAT的访问速度会影响音频流读取。确保你的音频文件是连续存储的复制进去后不要频繁删除、移动。对于超长音频考虑将其分割成多个短文件用NEXT触发器串联播放比读取一个巨大文件更流畅。一个实用的性能测试创建一个简单的测试脚本同时触发多个LATCH循环播放。从2个声音开始逐渐增加直到听到明显的卡顿或爆音。这个临界点就是你这套硬件和当前音频参数配置下的实际可用复音数。记录下来作为你项目设计的依据。6. 从原型到产品进阶应用与思考当基础功能跑通后我们可以思考如何将这个系统集成到更复杂的项目中或者进行功能扩展。1. 与其他传感器集成按钮只是触发方式的一种。你可以轻松地将触发源从keypad.Keys换成其他传感器。例如使用光线传感器analogio在环境变暗时触发恐怖音效或用加速度传感器adafruit_lis3dh在设备被拿起时播放欢迎语。只需将传感器读取的逻辑放入主循环在满足条件时直接调用对应trigger对象的on_press()方法即可模拟按钮按下。2. 动态加载音频文件当前系统在启动时扫描文件。但你可以修改代码实现运行时动态加载新的音频文件例如从通过网络下载或SD卡读取。这需要更复杂的文件管理和触发器动态创建逻辑但对于需要更新内容的应用如信息亭、展览很有用。3. 音量与混音控制代码中使用了audiomixer.Mixer。这个Mixer对象本身可以设置整体音量(mixer.volume)。你还可以在创建每个voice时设置其独立音量。例如在check_match_make_mixer函数创建mixer后你可以遍历mixer.voice列表为每个voice设置不同的初始音量级别让背景音乐比音效轻一些。4. 单声道版本的应用场景单声道版本(monophonic文件夹下的代码)虽然不能混音但它的优势在于对音频文件规格没有统一要求且代码更简单、资源占用更少。它完美适用于直接替换老式VS1000 Audio FX板卡的场景。如果你有一个旧项目里面存满了各种不同格式、采样率的OGG或WAV文件用这个版本几乎可以无缝迁移只需将文件转换为MP3即可。我个人在几个装置项目中实践下来的体会是这套系统的最大优势在于其声明式的配置方法。你把大部分逻辑都写在了文件名里而不是代码里。这使得非程序员如艺术家、设计师也能参与内容创作——他们只需要学会用Audacity处理和命名文件。而作为开发者我的工作就变成了搭建稳定的硬件平台和编写那一次性的核心框架代码。这种关注点分离让跨领域协作变得异常顺畅。最后一个小技巧如果你发现某个复杂的触发逻辑无法通过现有的5种触发器类型实现不要犹豫去自定义一个新的触发器类。这个框架的扩展性很好核心的TriggerBase类已经把资源管理和文件播放的脏活累活都干了你只需要在on_press和on_release里定义“何时播放”以及“播放哪个文件”的逻辑即可。这就像用乐高积木搭建一个专属的交互声音引擎其乐趣和可能性远不止于播放几个简单的音效。

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