基于STM32开发的智能语音助手系统
目录
- 引言
- 环境准备工作
- 硬件准备
- 软件安装与配置
- 系统设计
- 系统架构
- 硬件连接
- 代码实现
- 初始化代码
- 控制代码
- 应用场景
- 智能家居控制
- 个人语音助理
- 常见问题及解决方案
- 常见问题
- 解决方案
- 结论
1. 引言
随着人工智能技术的发展,智能语音助手已经逐渐进入了人们的日常生活。通过语音识别和自然语言处理,智能语音助手可以帮助用户完成各种任务,如控制家电、设置提醒和查询信息。本文将介绍如何使用STM32微控制器设计和实现一个基础的智能语音助手系统。
2. 环境准备工作
硬件准备
- STM32开发板(例如STM32F103C8T6)
- 音频输入模块(例如麦克风模块)
- 音频输出模块(例如扬声器模块)
- Wi-Fi模块(例如ESP8266,用于联网处理语音数据)
- OLED显示屏(用于显示系统状态)
- 按钮和LED(用于用户交互)
- 面包板和连接线
- USB下载线
软件安装与配置
- Keil uVision:用于编写、编译和调试代码。
- STM32CubeMX:用于配置STM32微控制器的引脚和外设。
- ST-Link Utility:用于将编译好的代码下载到STM32开发板中。
- 在线语音识别服务(例如Google Speech-to-Text API):用于语音数据处理。
步骤:
- 下载并安装Keil uVision。
- 下载并安装STM32CubeMX。
- 下载并安装ST-Link Utility。
- 注册并获取语音识别API的访问密钥。
3. 系统设计
系统架构
智能语音助手系统通过STM32微控制器连接麦克风、扬声器和Wi-Fi模块,实现语音采集、处理和反馈。语音数据通过Wi-Fi模块发送到在线语音识别服务进行处理,并根据识别结果控制设备或执行指令。系统包括语音采集模块、语音处理模块、设备控制模块和用户交互模块。
硬件连接
- 将麦克风模块的输出引脚连接到STM32的ADC引脚(例如PA0),VCC引脚连接到STM32的3.3V引脚,GND引脚连接到GND。
- 将扬声器模块的输入引脚连接到STM32的DAC引脚(例如PA4),VCC引脚连接到STM32的3.3V引脚,GND引脚连接到GND。
- 将Wi-Fi模块(例如ESP8266)的TX、RX引脚分别连接到STM32的USART引脚(例如PA9、PA10),VCC引脚连接到STM32的3.3V引脚,GND引脚连接到GND。
- 将OLED显示屏的VCC引脚连接到STM32的3.3V引脚,GND引脚连接到GND,SCL引脚连接到STM32的SCL引脚(例如PB6),SDA引脚连接到STM32的SDA引脚(例如PB7)。
- 将按钮的一个引脚连接到STM32的GPIO引脚(例如PA1),另一个引脚连接到GND。
- 将LED的正极引脚连接到STM32的GPIO引脚(例如PA2),负极引脚连接到GND。
4. 代码实现
初始化代码
#include "stm32f1xx_hal.h"
#include "wifi.h"
#include "audio_input.h"
#include "audio_output.h"
#include "oled.h"
#include "button.h"
#include "led.h"void SystemClock_Config(void);
static void MX_GPIO_Init(void);
static void MX_USART1_UART_Init(void);
static void MX_ADC1_Init(void);
static void MX_DAC_Init(void);
static void MX_I2C1_Init(void);int main(void) {HAL_Init();SystemClock_Config();MX_GPIO_Init();MX_USART1_UART_Init();MX_ADC1_Init();MX_DAC_Init();MX_I2C1_Init();WiFi_Init();AudioInput_Init();AudioOutput_Init();OLED_Init();Button_Init();LED_Init();while (1) {if (Button_IsPressed()) {OLED_DisplayString("Listening...");char audioData[512];AudioInput_Capture(audioData, sizeof(audioData));OLED_DisplayString("Processing...");char* result = WiFi_SendAudioForProcessing(audioData);OLED_DisplayString(result);if (strcmp(result, "Turn on the light") == 0) {LED_On();AudioOutput_Play("Light turned on");} else if (strcmp(result, "Turn off the light") == 0) {LED_Off();AudioOutput_Play("Light turned off");}HAL_Delay(1000);}}
}void SystemClock_Config(void) {// 配置系统时钟
}static void MX_GPIO_Init(void) {// 初始化GPIO__HAL_RCC_GPIOA_CLK_ENABLE();GPIO_InitTypeDef GPIO_InitStruct = {0};GPIO_InitStruct.Pin = GPIO_PIN_1 | GPIO_PIN_2;GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;GPIO_InitStruct.Pull = GPIO_NOPULL;GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);
}static void MX_USART1_UART_Init(void) {// 初始化USART1huart1.Instance = USART1;huart1.Init.BaudRate = 115200;huart1.Init.WordLength = UART_WORDLENGTH_8B;huart1.Init.StopBits = UART_STOPBITS_1;huart1.Init.Parity = UART_PARITY_NONE;huart1.Init.Mode = UART_MODE_TX_RX;huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE;huart1.Init.OverSampling = UART_OVERSAMPLING_16;if (HAL_UART_Init(&huart1) != HAL_OK) {Error_Handler();}
}static void MX_ADC1_Init(void) {// 初始化ADC1ADC_ChannelConfTypeDef sConfig = {0};hadc1.Instance = ADC1;hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE;hadc1.Init.ContinuousConvMode = ENABLE;hadc1.Init.DiscontinuousConvMode = DISABLE;hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;hadc1.Init.NbrOfConversion = 1;if (HAL_ADC_Init(&hadc1) != HAL_OK) {Error_Handler();}sConfig.Channel = ADC_CHANNEL_0;sConfig.Rank = ADC_REGULAR_RANK_1;sConfig.SamplingTime = ADC_SAMPLETIME_55CYCLES_5;if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK) {Error_Handler();}HAL_ADC_Start(&hadc1);
}static void MX_DAC_Init(void) {// 初始化DACDAC_ChannelConfTypeDef sConfig = {0};hdac.Instance = DAC;hdac.Init.DAC_Trigger = DAC_TRIGGER_NONE;hdac.Init.DAC_OutputBuffer = DAC_OUTPUTBUFFER_ENABLE;if (HAL_DAC_Init(&hdac) != HAL_OK) {Error_Handler();}sConfig.DAC_Trigger = DAC_TRIGGER_NONE;sConfig.DAC_OutputBuffer = DAC_OUTPUTBUFFER_ENABLE;if (HAL_DAC_ConfigChannel(&hdac, &sConfig, DAC_CHANNEL_1) != HAL_OK) {Error_Handler();}HAL_DAC_Start(&hdac, DAC_CHANNEL_1);
}static void MX_I2C1_Init(void) {// 初始化I2C1hi2c1.Instance = I2C1;hi2c1.Init.ClockSpeed = 100000;hi2c1.Init.DutyCycle = I2C_DUTYCYCLE_2;hi2c1.Init.OwnAddress1 = 0;hi2c1.Init.AddressingMode = I2C_ADDRESSINGMODE_7BIT;hi2c1.Init.DualAddressMode = I2C_DUALADDRESS_DISABLE;hi2c1.Init.OwnAddress2 = 0;hi2c1.Init.GeneralCallMode = I2C_GENERALCALL_DISABLE;hi2c1.Init.NoStretchMode = I2C_NOSTRETCH_DISABLE;if (HAL_I2C_Init(&hi2c1) != HAL_OK) {Error_Handler();}
}
控制代码
#include "wifi.h"
#include "audio_input.h"
#include "audio_output.h"
#include "oled.h"
#include "button.h"
#include "led.h"void WiFi_Init(void) {// 初始化Wi-Fi模块
}char* WiFi_SendAudioForProcessing(char *audioData) {// 发送音频数据到服务器进行处理// 返回处理结果
}void AudioInput_Init(void) {// 初始化音频输入模块
}void AudioInput_Capture(char *buffer, int length) {// 采集音频数据
}void AudioOutput_Init(void) {// 初始化音频输出模块
}void AudioOutput_Play(char *message) {// 播放音频提示
}void OLED_Init(void) {// 初始化OLED显示屏
}void OLED_DisplayString(char *str) {// 在OLED显示屏上显示字符串
}void Button_Init(void) {// 初始化按钮
}bool Button_IsPressed(void) {// 检测按钮是否按下
}void LED_Init(void) {// 初始化LED
}void LED_On(void) {// 打开LED
}void LED_Off(void) {// 关闭LED
}
⬇帮大家整理了单片机的资料
包括stm32的项目合集【源码+开发文档】
点击下方蓝字即可领取,感谢支持!⬇
点击领取更多嵌入式详细资料
问题讨论,stm32的资料领取可以私信!
5. 应用场景
智能家居控制
通过语音助手,用户可以使用自然语言控制家中的各类智能设备,如灯光、空调、电视等,实现便捷的智能家居体验。
个人语音助理
智能语音助手还可以用于个人语音助理,帮助用户设置提醒、查询信息,甚至执行一些简单的日常任务,如计算、转换单位等。
6. 常见问题及解决方案
常见问题
- 语音识别不准确
- Wi-Fi模块无法连接网络
- 音频播放效果不佳
解决方案
- 优化麦克风输入
- 选择高质量的麦克风模块,并调整音频采集参数,以提高语音识别的准确性。
- 检查Wi-Fi连接
- 确认Wi-Fi模块与STM32连接正常,确保网络配置正确。
- 调整音频输出
- 调整DAC输出参数或更换扬声器模块,确保音频播放的清晰度和音量。
7. 结论
本文介绍了如何使用STM32微控制器和多种模块实现一个智能语音助手系统,从硬件准备、环境配置到代码实现,详细介绍了每一步的操作步骤。通过本文的学习,读者可以掌握基本的嵌入式开发技能,并将其应用到人工智能项目中。
相关文章:

基于STM32开发的智能语音助手系统
目录 引言环境准备工作 硬件准备软件安装与配置系统设计 系统架构硬件连接代码实现 初始化代码控制代码应用场景 智能家居控制个人语音助理常见问题及解决方案 常见问题解决方案结论 1. 引言 随着人工智能技术的发展,智能语音助手已经逐渐进入了人们的日常生活。…...
基于python的图像去雾算法研究系统设计与实现
博主介绍: 大家好,本人精通Java、Python、C#、C、C编程语言,同时也熟练掌握微信小程序、Php和Android等技术,能够为大家提供全方位的技术支持和交流。 我有丰富的成品Java、Python、C#毕设项目经验,能够为学生提供各类…...
自定义 View 可以播放一段视频
请实现一个自定义 View 的核心代码,核心要求可以响应如下事件: // - 要求自定义 View 可以播放一段视频 / - 在 view 左侧区域上下滑动,可以提高减少音量 / / - 在 view 右侧区域上下滑动可以提高减少屏幕亮度 // - 在 view 左右滑动可以…...

LVS负载均衡集群部署之—NAT模式的介绍及搭建步骤
一、环境准备 1.准备三台rhel9服务器 服务器名称 主机名 ip地址备注LVS调度服务器lvs.timinglee.org eth0:172.25.254.100(外网) eth1:192.168.0.100(内网) 关闭selinux和防火墙webserver2网站服务器webserver1.timinglee.orgeth0:192.168.…...
【算法】浅析哈希算法【附代码示例】
哈希算法:数据存储与检索的基石 1. 引言 在计算机科学中,哈希算法是一种用于将数据(如文件、网络地址或数据库记录)转换为固定长度的哈希值的过程。哈希值通常是一个较短的数字或字符串,用于快速检索原始数据或验证数…...

2024.8.12
2024.8.12 【梦最让我费解的地方在于,明明你看不清梦里人们的脸,却清晰地知道他们是谁。】 Monday 七月初九 序理论 最小链覆盖&最长反链长度 我们设定一个二元关系符R和一个集合A 我们设定<A,R>这样一个类群,那么对于任意 a i…...
使用Python解析pdf、docx等格式文件。
针对不同类型的文件,需要采取特定的访问与解析策略来有效获取其中蕴含的知识。下面我们将介绍对于不同数据源数据的获取方式。 1 解析Docx文档 1.1 获取Docx文档中文本 from docx import Document # pip install python-docx # python-docx 0.8.11 filename xx…...
Linux网络通信基础API
这篇文章只有Linux网络通信基础API大参数信息,和返回值,这篇文章并没有这些基础API的参数类型介绍。accept的第二个参数可以查看客户端信息。 创建socket #include <sys/types.h> /* See NOTES */#include <sys/socket.h>int sock…...

Python爬虫:下载4K壁纸
🎁🎁创作不易,关注作者不迷路🎀🎀 目录 🌸完整代码 🌸分析 🎁基本思路 🎁需要的库 🎁提取图片的链接和标题 👓寻找Cookie和User-Agent &…...

2024年【北京市安全员-B证】新版试题及北京市安全员-B证免费试题
题库来源:安全生产模拟考试一点通公众号小程序 2024年【北京市安全员-B证】新版试题及北京市安全员-B证免费试题,包含北京市安全员-B证新版试题答案和解析及北京市安全员-B证免费试题练习。安全生产模拟考试一点通结合国家北京市安全员-B证考试最新大纲…...

python爬取B站视频实验
实验17:爬虫2 文章目录 实验17:爬虫21.实验目标及要求2. 实验主要内容3.实验小结 1.实验目标及要求 (1)掌握有关爬虫的包 (2)掌握爬虫方法 (3)爬取B站卡塔尔世界杯若干视频 2. 实验…...

10步搞定Python爬虫从零到精通!
学习Python网络爬虫可以分为以下几个步骤,每一步都包括必要的细节和示例代码,以帮助你从零开始掌握这一技能。 第一步:理解网络爬虫基础 什么是网络爬虫? 网络爬虫是一种自动化程序,用来从互联网上收集数据.它通过发送 HTTP 请求…...

SpringMVC学习笔记---带你快速入门和复习
一、初识SpringMVC 1.1、什么是SpringMVC 1.1.1、什么是MVC MVC是一种软件架构模式(是一种软件架构设计思想,不止Java开发中用到,其它语言也需要用到),它将应用分为三块: M:Model࿰…...

Linux系统编程 day09 线程同步
Linux系统编程 day09 线程同步 1.互斥锁2.死锁3.读写锁4.条件变量(生产者消费者模型)5.信号量 1.互斥锁 互斥锁是一种同步机制,用于控制多个线程对共享资源的访问,确保在同一时间只有一个线程可以访问特定的资源或执行特定的操作…...

Vue快速入门(四)——Vue3及组合式API(一)
文章目录 一、认识Vue31. Vue2 选项式 API vs Vue3 组合式API2. Vue3的优势二、使用create-vue搭建Vue3项目1、认识create-vue2. 使用create-vue创建项目三、组合式API - setup选项1. setup选项的写法和执行时机2. setup中写代码的特点3. `<script setup>`语法糖4.小结四…...

vue项目名修改、webstorm和idea创建的项目重命名、重构项目、修改项目名称
一、需求 就是创建了一个项目,后期需要重命名,怎么办?----> 直接修改?肯定不行,因为里面有些配置也需要修改,假如你只改文件夹名称的话,里面配置都没修改,后期可能会出问题。 二…...

【MySQL】数据库约束和多表查询
目录 1.前言 2.数据库约束 2.1约束类型 2.2 NULL约束 2.3 NUIQUE:唯一约束 2.4 DEFAULT:默认值约束 2.5 PRIMARY KEY:主键约束 2.6 FOREIGN KEY:外键约束 1.7 CHECK约束 3.表的设计 3.1一对一 3.2一对多 3.3多对多 …...

抖店飞鸽客服自动回复软件开发教程与下载体验(.NET版)
转载请注明出处! 原文链接:https://blog.csdn.net/zgyulongfei/article/details/140960430 本文适合的读者为: 抖店(抖音小店)个体商家;抖店店群商家(店群商家:指的是开了几十个抖…...
如何关闭redis的自动清理缓存,声明式事务(含有redis)如何解决,redis setnx锁的使用。
20240809 一、解决redis数据被删除的方案1、发现问题2、解决注意!! 二、声明式事务(当有redis的时候)1. 先看代码2. Transactional(rollbackFor Exception.class)3. 如何解决redis在事务里面,如何保证原子性和一致性3…...
C#中抽象类的使用
前言 我们在C#中使用抽象类可以发挥C#多态的功能,把具有共性的方法定义在抽象类中,然后在不同的类中去实现,可增强代码的可读性、扩展性。 1、不使用抽象类 我们定义了下面两个类XiaoWang、XiaoMing,他们有一个Country方法&…...
基于算法竞赛的c++编程(28)结构体的进阶应用
结构体的嵌套与复杂数据组织 在C中,结构体可以嵌套使用,形成更复杂的数据结构。例如,可以通过嵌套结构体描述多层级数据关系: struct Address {string city;string street;int zipCode; };struct Employee {string name;int id;…...

YSYX学习记录(八)
C语言,练习0: 先创建一个文件夹,我用的是物理机: 安装build-essential 练习1: 我注释掉了 #include <stdio.h> 出现下面错误 在你的文本编辑器中打开ex1文件,随机修改或删除一部分,之后…...

跨链模式:多链互操作架构与性能扩展方案
跨链模式:多链互操作架构与性能扩展方案 ——构建下一代区块链互联网的技术基石 一、跨链架构的核心范式演进 1. 分层协议栈:模块化解耦设计 现代跨链系统采用分层协议栈实现灵活扩展(H2Cross架构): 适配层…...
CMake控制VS2022项目文件分组
我们可以通过 CMake 控制源文件的组织结构,使它们在 VS 解决方案资源管理器中以“组”(Filter)的形式进行分类展示。 🎯 目标 通过 CMake 脚本将 .cpp、.h 等源文件分组显示在 Visual Studio 2022 的解决方案资源管理器中。 ✅ 支持的方法汇总(共4种) 方法描述是否推荐…...

Reasoning over Uncertain Text by Generative Large Language Models
https://ojs.aaai.org/index.php/AAAI/article/view/34674/36829https://ojs.aaai.org/index.php/AAAI/article/view/34674/36829 1. 概述 文本中的不确定性在许多语境中传达,从日常对话到特定领域的文档(例如医学文档)(Heritage 2013;Landmark、Gulbrandsen 和 Svenevei…...
CSS设置元素的宽度根据其内容自动调整
width: fit-content 是 CSS 中的一个属性值,用于设置元素的宽度根据其内容自动调整,确保宽度刚好容纳内容而不会超出。 效果对比 默认情况(width: auto): 块级元素(如 <div>)会占满父容器…...
音视频——I2S 协议详解
I2S 协议详解 I2S (Inter-IC Sound) 协议是一种串行总线协议,专门用于在数字音频设备之间传输数字音频数据。它由飞利浦(Philips)公司开发,以其简单、高效和广泛的兼容性而闻名。 1. 信号线 I2S 协议通常使用三根或四根信号线&a…...

C/C++ 中附加包含目录、附加库目录与附加依赖项详解
在 C/C 编程的编译和链接过程中,附加包含目录、附加库目录和附加依赖项是三个至关重要的设置,它们相互配合,确保程序能够正确引用外部资源并顺利构建。虽然在学习过程中,这些概念容易让人混淆,但深入理解它们的作用和联…...
C#学习第29天:表达式树(Expression Trees)
目录 什么是表达式树? 核心概念 1.表达式树的构建 2. 表达式树与Lambda表达式 3.解析和访问表达式树 4.动态条件查询 表达式树的优势 1.动态构建查询 2.LINQ 提供程序支持: 3.性能优化 4.元数据处理 5.代码转换和重写 适用场景 代码复杂性…...

Git 3天2K星标:Datawhale 的 Happy-LLM 项目介绍(附教程)
引言 在人工智能飞速发展的今天,大语言模型(Large Language Models, LLMs)已成为技术领域的焦点。从智能写作到代码生成,LLM 的应用场景不断扩展,深刻改变了我们的工作和生活方式。然而,理解这些模型的内部…...