AI智能语音识别模块(二)——基于Arduino的语音控制MP3播放器

news2024/11/28 23:56:12

文章目录

  • 简介
  • 离线语音控制模块
  • Mini MP3模块
  • 0.96寸 OLED模块
  • 实验准备
  • 安装库
  • 接线定义
  • 主要程序
  • 实验效果
  • 注意事项
  • 总结

简介

在前面一篇文章里我们对AI智能语音识别模块进行了介绍,并对离线语音模组下载固件的过程进行了一个简单描述,不知道大家还记不记得,这篇文章也是鸽了好久,,本文将用这个语音控制模块结合前面介绍的DFPlayer Mini MP3模块来做一个有趣的应用,在上一期文章中,我们只是简单的做了一个只用语音控制模块点亮LED的应用,本期将结合Arduino开发板、DFPlayer Mini MP3模块来实现对音乐的控制,并在OLED屏幕显示基本的控制信息。本期是基于前面两篇文章进行的实验,因此就不对语音模块下载固件的过程和MP3模块的引脚定义进行介绍了。
还没看过这两篇文章的可以点击下面两个链接跳转:

添加AI智能语音识别模块(一)——离线模组介绍:

Arduino驱动DFPlayer Mini MP3模块:

本次实验用到的主要器件包含下面三个模块:

离线语音控制模块

在这里插入图片描述

Mini MP3模块

在这里插入图片描述

0.96寸 OLED模块

在这里插入图片描述

实验准备

Arduino开发板1个
Mini MP3模块1个
离线语音模块1个
TF卡1个
8欧1W喇叭1个
0.96寸OLED1个
数据线(用于板子下载程序和供电)2条
杜邦线若干

安装库

程序中使用到了DFRobotDFPlayerMini.h头文件,如果没有安装库编译代码时会报错,因此需要正确安装DFRobotDFPlayerMini库才能编译通过。下面是安装介绍。
在这里插入图片描述
在这里插入图片描述

接线定义

Arduinodfplayer
5VVCC
GNDGND
D10TXD
D11RXD
ArduinoOLED
GNDGND
3.3VVCC
A5SCL
A4SDA
Arduino语音识别模块
D0B7

主要程序

/*
 功能: 通过语音控制MP3模块的播放、暂停、上一曲、下一曲、增大音量、减小音量等,并在OLED屏幕显示控制信息。   
*/

#include "SoftwareSerial.h"
#include "DFRobotDFPlayerMini.h"
#include "config.h"

char uart_rx_buf = 0; 
SoftwareSerial  mySoftwareSerial(10, 11); // RX, TX

DFRobotDFPlayerMini myDFPlayer;

void setup()
{   
  mySoftwareSerial.begin(9600);
  Serial.begin(9600);
  uint8_t t=' ';
  OLED_Init();
  OLED_ColorTurn(0);//0正常显示 1反色显示
  OLED_DisplayTurn(0);//0正常显示 1翻转180度显示
  
  OLED_ShowChar(24,0,'M',16);
  OLED_ShowChar(42,0,'U',16);
  OLED_ShowChar(60,0,'S',16);
  OLED_ShowChar(78,0,'I',16);
  OLED_ShowChar(96,0,'C',16);
  myDFPlayer.begin(mySoftwareSerial);
  
  myDFPlayer.volume(20);   //音量设定为20 
  
  myDFPlayer.enableLoopAll() ;   
 
}

void loop()
{
     if(Serial.available()>0) {  //如果串口有数据
            
        uart_rx_buf = char(Serial.read());//Serial.read()读缓冲区字符,每次只能读一个字节
        if((uart_rx_buf&0xf0) >>4 == 2) {
         switch(uart_rx_buf & 0x0f){
         
           case 0:
                    myDFPlayer.start();
                    delay (500);
                    Serial.println("start");   
                    
                    OLED_ShowChinese(0,3,4,16);
                    OLED_ShowChinese(18,3,5,16);
                    OLED_ShowChinese(36,3,6,16);
                    OLED_ShowChinese(54,3,7,16); 
            break;

            case 1:
                   myDFPlayer.pause();
                   delay (500);
                   Serial.println("pause"); 
                   
                   OLED_ShowChinese(0,3,8,16);
                   OLED_ShowChinese(18,3,9,16);
                   OLED_ShowChinese(36,3,4,16);
                   OLED_ShowChinese(54,3,5,16); 
            break;                
          }
        } 
       if((uart_rx_buf&0xf0) >>4 == 3) {
         switch(uart_rx_buf & 0x0f){         
           case 0:
                     myDFPlayer.previous() ;
                    delay (500);
                    Serial.println("previous"); 
           
                    OLED_ShowChinese(0,3,0,16);
                    OLED_ShowChinese(18,3,1,16);
                    OLED_ShowChinese(36,3,2,16);
                    OLED_ShowChinese(54,3,10,16);
            break;

            case 1:
             myDFPlayer.next() ;
                  delay (500);
                  Serial.println("next"); 
          
               OLED_ShowChinese(0,3,3,16);
               OLED_ShowChinese(18,3,1,16);
               OLED_ShowChinese(36,3,2,16);
               OLED_ShowChinese(54,3,10,16);
            break;                
          }
        }             

          if((uart_rx_buf&0xf0) >>4 == 4) {
         switch(uart_rx_buf & 0x0f){         
           case 0:
                     myDFPlayer.volumeUp();
                     delay (500);
                     Serial.println("volumeUp"); 
                     
                     OLED_ShowChinese(0,3,11,16);
                     OLED_ShowChinese(18,3,12,16);
                     OLED_ShowChinese(36,3,6,16);
                     OLED_ShowChinese(54,3,13,16);
            break;

            case 1:
                     myDFPlayer.volumeDown()  ;
                     delay (500);
                     Serial.println("volumeDown");
                      
                     OLED_ShowChinese(0,3,14,16);
                     OLED_ShowChinese(18,3,15,16);
                     OLED_ShowChinese(36,3,6,16);
                     OLED_ShowChinese(54,3,13,16);
            break;                
          }
        }           
        if((uart_rx_buf&0xf0) >>4 == 5) {
         switch(uart_rx_buf & 0x0f){         
           case 0:
                     myDFPlayer.volume(30) ;
                     delay (500);
                     Serial.println("MaxVolume"); 
                     OLED_ShowChinese(0,3,16,16);
                     OLED_ShowChinese(18,3,12,16);
                     OLED_ShowChinese(36,3,6,16);
                     OLED_ShowChinese(54,3,13,16);
            break;

            case 1:
                     myDFPlayer.volume(20) ;
                     delay (500);
                     Serial.println("MidVolume"); 
                     OLED_ShowChinese(0,3,17,16);
                     OLED_ShowChinese(18,3,18,16);
                     OLED_ShowChinese(36,3,6,16);
                     OLED_ShowChinese(54,3,13,16);
            break;   

            case 2:
                     myDFPlayer.volume(15) ;
                     delay (500);
                     Serial.println("MinVolume"); 
                     OLED_ShowChinese(0,3,16,16);
                     OLED_ShowChinese(18,3,15,16);
                     OLED_ShowChinese(36,3,6,16);
                     OLED_ShowChinese(54,3,13,16);
            break;   
          }
        }           
       
    uart_rx_buf = 0;            //清除接收到的字符
   }
}

查看上面代码,主要是对串口接收到的数据进行处理,下面贴出了离线语音模块的控制详情。对串口发送的数据是可以自定义的,因此对上述代码稍加修改就可以只用一个Switch语句或者多个if语句实现串口控制,这里采用先判断串口高4位数据,然后再判断低4位数据的方法来实现控制。这个其实和前面一期Arduino驱动DFPlayer Mini MP3模块的文章一样,只不过对串口输出的判断,从判断高低电平变为判断接收到的串口数据。这样只用一个端口就可以实现控制,节省了IO口数目。除了对接收到的数据进行判断处理的代码,还有oled显示控制信息的代码,这个就没什么好介绍的了,大家也可以自己丰富关于显示这部分的功能。

上述代码使用了两个串口,D10,D11是Arduino和MP3播放器通信使用的,另一个是和语音模块通信用的,使用到了Arduino的硬件串口,因此这里要注意在下载程序时把接到 D0 引脚的线拔掉,否则程序会下载失败。另一个解决办法是使用软件串口,这样就避免了串口冲突。
在这里插入图片描述

在这里插入图片描述
前面我们已经讲过AI智能语音模块下载固件的过程了,这里就不再重复了,烧录好固件后,可以说以下命令词实现控制。通过下面三种唤醒词唤醒模块。
在这里插入图片描述
通过下面几个命令词实现控制MP3音乐模块:
在这里插入图片描述
在说命令词的时候可能会受到周围环境干扰或者喇叭播放音乐时自身太吵的影响,导致识别不到或者识别错误指令,这个没关系多试几遍就好了,一般不是很吵的情况下说第二遍就能识别正确了,这个语音模块识别正确率还是很高的。本次实验在固件生成的时候用到的是快速测试版本,可以快速生成SDK进行测试,该版本识别的效果相对发布版本语音识别效果差一些,因为本次实验只是为了验证功能的实现,如果想要更稳定的版本,大家可以自行生成测试一下。

实验效果

整体接线:

在这里插入图片描述
控制效果:
在这里插入图片描述

基于Arduino的语音控制MP3播放器

注意事项

  1. 使用本次实验的语音控制模块,下载离线语音模块固件时,需要断开电源后重新通电,才能下载成功。
  2. Arduino上传上述代码时,需要拔插Arduino的 D0 引脚的线,否则上传失败。
  3. 可以使用功率更大一点的喇叭以提升音质效果,但要在3W以内。

总结

本次实验通过Arduino,结合语音模块和MP3模块以及oled模块,实现了对MP3模块的语音控制和显示控制信息功能。

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

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

相关文章

Nginx+keepalived实现七层的负载均衡

1.keepalived VRRP 介绍 keepalived是什么? keepalived是集群管理中保证集群高可用的一个服务软件,用来防止单点故障。 keepalived工作原理 keepalived是以VRRP协议为实现基础的,VRRP全称Virtual Router Redundancy Protocol&…

视频AI智剪,批量剪辑助力高效创作

视频AI智剪是一种基于人工智能技术的自动化剪辑工具,它可以自动对视频素材进行分析、筛选、剪辑和优化,从而生成一部高质量的视频作品。而批量剪辑则是指利用AI智剪技术,同时对多个视频素材进行自动化剪辑,大大提高了剪辑效率。本…

NI‑9237国产化50 kS/s/ch,桥模拟输入,4通道C系列应变/桥输入模块

50 kS/s/ch,桥模拟输入,4通道C系列应变/桥输入模块 NI‑9237提供了所有的信号调理功能来实现多达四个基于桥的传感器的供电和测量。该模块提供通道间零相位延迟的应变或负载测量。它还具有60 VDC隔离和1,000 Vrms瞬态隔离,提供高…

操作系统 day03(运行机制)

机器指令 二进制机器指令就是处理器(CPU)能识别、执行的最基本命令 程序运行的过程就是CPU执行一条一条的机器指令的过程 应用程序和内核程序 操作系统的最重要角色是:系统资源的管理者,而操作系统的对系统资源的管理工作就是…

Mysql系列 -索引模型数据结构

索引就是排好序的数据结构,可以帮助我们快速的查找到数据,那么底层的数据到底是如何存储的呢? 为什么InnoDB 用的是Btree 存储结构? 大家可以看看这个可视化的网站 数据结构和算法的可视化工具 可以看到数据结构里面有链表&…

JVM虚拟机:垃圾回收算法和垃圾回收器之间的关系

GC垃圾回收算法 在前面的课程中我们学习了GC垃圾回收算法,分别为: 引用回收算法 复制算法 标记清除算法 标记整理算法 这些垃圾回收算法是理论,有多种垃圾回收器可以实现这些理论。目前为止没有最完美的垃圾回收器,只能针对具体的情况选择最合适的垃圾回收器,进行分代收集…

高校教务系统登录页面JS分析——天津大学

高校教务系统密码加密逻辑及JS逆向 本文将介绍天津大学教务系统的密码加密逻辑以及使用JavaScript进行逆向分析的过程。通过本文,你将了解到密码加密的基本概念、常用加密算法以及如何通过逆向分析来破解密码。 本文仅供交流学习,勿用于非法用途。 一、密…

MFC网络通信-Udp服务端

目录 1、UI的布局 2、代码的实现: (1)、自定义的子类CServerSocket (2)、重写OnReceive事件 (3)、在CUdpServerDlg类中处理 (4)、在OnInitDialog函数中 &#xff0…

Leetcode刷题详解——Pow(x, n)

1. 题目链接:50. Pow(x, n) 2. 题目描述: 实现 pow(x, n) ,即计算 x 的整数 n 次幂函数(即,xn )。 示例 1: 输入:x 2.00000, n 10 输出:1024.00000示例 2:…

Mozilla Firefox 119 现已可供下载

Mozilla Firefox 119 开源网络浏览器现在可以下载了,是时候先看看它的新功能和改进了。 Firefox 119 改进了 Firefox View 功能,现在可以提供更多内容,如最近关闭的标签页和浏览历史,你可以按日期或网站排序,还支持查…

项目知识点总结-住房图片信息添加-Excel导出

(1)住房信息添加 Controller: RequestMapping("/add")public String add(Home home, Model model) throws IOException{String sqlPath null;//定义文件保存的本地路径String localPath"D:\\AnZhuang\\Java项目\\选题\\Xin-…

YOLOv5 分类模型的加载

YOLOv5 分类模型的加载 flyfish 版本 6.2 yolov5s分类模型 python classify/train.py --model resnet18.pt --data cifar100 --epochs 5 --img 224resnet18模型 python classify/train.py --model resnet18.pt --data cifar100 --epochs 5 --img 128导出模型看一下结构 p…

基于SSM的理发店管理系统

基于SSM的理发店管理系统的设计与实现~ 开发语言:Java数据库:MySQL技术:SpringSpringMVCMyBatis工具:IDEA/Ecilpse、Navicat、Maven 系统展示 主页 公告信息 管理员界面 用户界面 摘要 基于SSM(Spring、Spring MVC、…

水库大坝可视化智能远程监管方案,助力安全监测智能巡检

一、背景需求 水库大坝作为防洪度汛的重要设施,其安全问题直接关系到人民群众的生命财产安全。因此,必须加强对大坝水库的安全管理,对水库除险加固和运行管护要消除存量隐患,实现常态化管理,同时要配套完善重点小型水…

windows使用FindWindow函数查找窗口句柄

理解什么是句柄? 对于“句柄”,之前一直停留在一知半解的认识层面,也说不清具体概念,只知道它是一个标识符,用来标记对象或者说某个东西的。只知其名不知其意。目前学习windows编程,对“句柄”做一个完整的…

GPT与人类共生:解析AI助手的兴起

随着GPT模型的崭新应用,如百度的​1​和CSDN的​2​,以及AI助手的普及,人们开始讨论AI对就业市场和互联网公司的潜在影响。本文将探讨GPT和AI助手的共生关系,以及我们如何使用它们,以及使用的平台和动机。 GPT和AI助手…

【AI视野·今日Robot 机器人论文速览 第六十一期】Tue, 24 Oct 2023

AI视野今日CS.Robotics 机器人学论文速览 Tue, 24 Oct 2023 Totally 50 papers 👉上期速览✈更多精彩请移步主页 Daily Robotics Papers Robot Fine-Tuning Made Easy: Pre-Training Rewards and Policies for Autonomous Real-World Reinforcement Learning Autho…

常用编程语言排行与应用场景汇总(2023.10)

文章目录 编程语言排行一、Python二、C三、C四、Java五、C#六、JavaScript七、VB(Visual Basic)八、PHP九、SQL十、ASM(Assembly Language)十一、Go十二、Scratch十三、Delphi/Object Pascal十四、MATLAB十五、Swift十六、Fortran…

十九、类型信息(5)

动态代理 _代理_是基本的设计模式之一。一个对象封装真实对象,代替其提供其他或不同的操作—这些操作通常涉及到与“真实”对象的通信,因此代理通常充当中间对象。这是一个简单的示例,显示代理的结构: interface Interface {voi…

【Java 进阶篇】Java Web开发:实现验证码功能

在Web应用程序中,验证码(CAPTCHA)是一种常见的安全工具,用于验证用户是否为人类而不是机器。验证码通常以图像形式呈现,要求用户在登录或注册时输入正确的字符。在这篇文章中,我们将详细介绍如何在Java Web…