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

通信原理课设(gec6818) 007:语音识别

目录

1、去科大讯飞官网下载对应的sdk

2、科大讯飞文件夹的意思

3、配置ARM的录音环境

4、编程实现语音识别


我们的需求是将一个语音文件从客户端传到服务器,因此我们最好是选用tcp
    
现在市面上面常用的语音识别解决方案为:科大讯飞c和百度c++

离线语音识别和在线语音识别有一定区别,以识别“你好”为例:
在线:语音识别可以直接返回“你好”这个字符串    
离线:只能做命令词 --- 根据我的需求写好代号
           如:你好 --- 1001
           最终我们可以拿到这个1001这个id

语音解决方案没有arm的环境,因此我们需要用开发板去录音,将这个录音文件发送给乌班图,由乌班图识别出来之后返回给arm板

1、去科大讯飞官网下载对应的sdk

进入科大讯飞官网 -> 讯飞开放平台 -> 离线命令词识别 -> 免费试用
    -> 下载对应的SDK-> 跳转到对应的sdk下载界面->注意下面三个红色框框的选择(下载sdk可能会提示要实名认证,认证就好了)

2、科大讯飞文件夹的意思

每个人下载的sdk都是不一样的

进入到\samples\asr_offline_sample,可以看到有一个makefile文件,我们在编译的时候直接make就可以了。make会找Makefile去执行,make完成如果没有报错,它会在bin目录里面生成一个asr_offline_sample可执行文件。

由于我们只用x64的库,所以我们用记事本打开makefile,将x86改成x64

在执行的时候要找到这个库  我们需要将这个库弄到 /lib文件夹

 sudo cp lib/x64/libmsc.so /lib

3、配置ARM的录音环境

链接:https://pan.baidu.com/s/1LZ0Lpj9DhqpMzhzYjp0xVg?pwd=scrt 
提取码:scrt

1、首先将alsa-1.0.tar.gz这个文件拷贝到开发板的 /home
     cp /mnt/udisk/alsa-1.0.tar.gz /home

2、在home目录里面解压这个文件
     tar xvf alsa-1.0.tar.gz

3、进入  alsa-1.0这个文件夹 
     cd  alsa-1.0
     进入这个文件夹里面的bin目录
     cd bin
     将这个bin目录里面的  arecord aplay这两个文件复制到 /bin
     cp arecord aplay /bin
    
4、 然后cd ..
     进入lib这个文件夹 cd bin
     将里面so这些动态库复制到 /lib这个文件夹
     cp libasound.s* /lib

5、回到home目录
      cd /home
      在home目录里面建立一个文件夹 叫gec
      mkdir gec
      继续在gec文件夹里面建立一个文件夹alsa-1.0.22
      mkdir -p /home/gec/alsa-1.0.22
      将这个alsa-1.0整体copy 到gec文件夹里面去 并且要换一个名字叫alsa-1.0.22(必须是这个名字)
      cd /home/alsa-1.0  进入这个文件夹
      cp -rf * /home/gec/alsa-1.0.22

6、录音环境配置完成,进行测试

      录音:arecord 1.wav  (ctrl + c结束)
      放音:aplay 1.wav

      录音4s,频率16000,保存为hehe.pcm:arecord -d4 -c1 -r16000 -traw -fS16_LE hehe.pcm
      放音:aplay -d4 -c1 -r16000 -traw -fS16_LE hehe.pcm

(录音是arecord ,放音是aplay ,其他参数都是一样的)

4、编程实现语音识别

我们进行语音识别时,也是在网络编程,需要客户端和服务端。在这里,客户端是开发板,开发板进行录音,并将录音文件发送给乌班图。Ubuntu是服务端,接收开发板发送过来的录音文件,并进行语音识别,返回语句的id。

在第二部分我们知道文件夹中的bin存放可执行文件以及识别的音频。我们进入bin文件夹可以看到一个call.bnf,用记事本打开:

看到这个,我们就能知道语音识别仅仅能识别返回在call.bnf定义了id的语句,那怎么才能识别我们想要说的话呢?简单,我们自己在里面加就可以了。比如加个“打开蜂鸣器”:

有了这个基础,我们后面才能实现语音控制开发板。

现在先实现简单的语句识别,即开发板录音并将录音文件发给ubuntu,ubuntu进行语音识别,并返回对应语句的id:

建立两个文件夹:client和server

client文件夹存放客户端程序tcp_client.c

server文件夹存放科大讯飞的sdk和服务端程序tcp_sever.c

由于这次是传文件,和上一篇网络编程的传法还是有些不同的

关键代码:

tcp_client.c :

void function(void)
{unsigned char buf[1024] = {0};while(1){//首先发送文件大小//阻塞你按回车printf("按回车继续\n");getchar();//弄你的文件printf("请录音4秒........\n");//获取音频文件 system("arecord -d4 -c1 -r16000 -traw -fS16_LE hehe.pcm");int fd = open("hehe.pcm",O_RDWR);if(-1 == fd){perror("open pcm error");exit(10);}int filesize = lseek(fd,0x00,SEEK_END);lseek(fd,0x00,SEEK_SET);//偏移到开头send(sockfd,&filesize,4,0);//接收信息 "error!!!" or "next!!!!"recv(sockfd,buf,9,0);printf("11111 %s\n",buf);if(strcmp(buf,"next!!!!")){printf("服务器错误了\n");continue;}//如果是"next!!!!"发送文件while(1){int r = read(fd,buf,1024);if(-1 == r){perror("read pcm error");break;}else if(0 == r){printf("over\n");break;				}else{send(sockfd,buf,r,0);}}close(fd);//等待接收idint id;recv(sockfd,&id,4,0);	if(id == 6666){printf("打印");}printf("id ===== %d\n",id);}
}

asr_offine_sample.c :

//一个全局的科大讯飞的id  也就是我最终想要得到的结果
int FlayId = 0;//0代表一个错误值//解析出相应的id出来 id固定为4位
int StringToId(const char * str)
{int id = 0;printf("------> %s\n",str);int len = strlen(str) - 3;//固定匹配  "id="这个字符串  模式匹配用正则表达式for(int i = 0;i < len;i++){if(!strncmp(str,"id=",3)){str += 4;//id=" 这个字符串给过掉printf("------> %s\n",str);id = (str[0]-'0')*1000+(str[1]-'0')*100+(str[2]-'0')*10+(str[3]-'0'); break;}str++;//一旦没有匹配 那么我们就往后面走一个}return id;
}//语音识别  返回结果
//返回的是ID  返回0表示识别失败
int GetFlayId(void)
{int ret = run_asr(&asr_data);if(MSP_SUCCESS != ret)//识别出错{printf("离线语法识别出错: %d \n", ret);return 0;}return FlayId;
}

tcp_server.c :

void SaveFile(int accceptfd,int filesize)
{//每一次都是重复的覆盖hehe.pcmint fd = open("wav/hehe.pcm",O_RDWR | O_TRUNC | O_CREAT, 0664);//截短这个文件if(-1 == fd){send(accceptfd,"error!!!",9,0);//失败发送这个错误return;}send(accceptfd,"next!!!!",9,0);//发送下一步的指令unsigned char buf[1024] = {0};int size = 0;//接收文件的内容while(1)		{int r = recv(accceptfd,buf,1024,0);if(-1 == r){perror("recv error");break;}else if(0 == r)//客户端已经断了{printf("对方断开连接了\n");break;}else//接收到信息了{//将文件的内容写入到文件write(fd,buf,r);//做完之后要退出size += r;if(size >= filesize)break;}}close(fd);
}//专门用于去服务一个客户的线程
void * ClinetFunction(void * arg)
{pthread_detach(pthread_self());//将其分离int * accceptfd = (int *)arg;printf(" * accceptfd = %d\n", * accceptfd);int filesize = 0;//你发什么信息过来  我就在这个信息之前加上一节 然后回发给你while(1){printf("\t\t等待客户端传文件过来........\n");int r = recv(*accceptfd,&filesize,4,0);//阻塞等待数据过来if(-1 == r){perror("recv error");break;}else if(0 == r)//客户端已经断了{printf("对方断开连接了\n");break;}else//接收到信息了{//文件大小SaveFile(* accceptfd,filesize);int id = GetFlayId();//文件接收完毕  那么我们就放过去识别即可//给客户端返回idsend(* accceptfd,&id,4,0);}}close(*accceptfd);free(accceptfd);return NULL;
}

完整工程:

链接:https://pan.baidu.com/s/1thUvAArWzcqmOT6QrvGHew?pwd=yuyi 
提取码:yuyi

相关文章:

  • 【算法题】30. 串联所有单词的子串
  • 如何使用ArcGIS Pro将Excel表转换为SHP文件
  • 14.用户管理
  • CentOS 5/6/7 基于开源项目制作openssh 9.6p1 rpm包—— 筑梦之路
  • 小程序wx:if 和hidden的区别?
  • Spring Data Redis对象缓存序列化问题
  • K8S Helm 安装ingress-nginx/ingress-nginx
  • 阿里云ECS云服务器优势整理(共9点)
  • LabVIEW在大型风电机组状态监测系统开发中的应用
  • 视频格式网络地址转换视频到本地,获取封面、时长,其他格式转换成mp4
  • 每日一题——LeetCode976
  • Python 下载与安装
  • C#,入门教程(01)—— Visual Studio 2022 免费安装的详细图文与动画教程
  • 2023 年中国金融级分布式数据库市场报告:TiDB 位列领导者梯队,创新能力与增长指数表现突出
  • 使用nodejs定时备份mysql数据库与恢复
  • 【干货分享】SpringCloud微服务架构分布式组件如何共享session对象
  • ABAP的include关键字,Java的import, C的include和C4C ABSL 的import比较
  • CSS相对定位
  • Essential Studio for ASP.NET Web Forms 2017 v2,新增自定义树形网格工具栏
  • java B2B2C 源码多租户电子商城系统-Kafka基本使用介绍
  • JavaScript-Array类型
  • magento2项目上线注意事项
  • python大佬养成计划----difflib模块
  • React系列之 Redux 架构模式
  • Spring核心 Bean的高级装配
  • SQLServer之创建显式事务
  • Terraform入门 - 1. 安装Terraform
  • use Google search engine
  • vue中实现单选
  • 简单易用的leetcode开发测试工具(npm)
  • 聊聊flink的BlobWriter
  • 每天10道Java面试题,跟我走,offer有!
  • 前端js -- this指向总结。
  • 前端技术周刊 2019-01-14:客户端存储
  • 让你的分享飞起来——极光推出社会化分享组件
  • 树莓派 - 使用须知
  • 突破自己的技术思维
  • 小试R空间处理新库sf
  • 用Visual Studio开发以太坊智能合约
  • 《TCP IP 详解卷1:协议》阅读笔记 - 第六章
  • postgresql行列转换函数
  • ​secrets --- 生成管理密码的安全随机数​
  • ​力扣解法汇总946-验证栈序列
  • #快捷键# 大学四年我常用的软件快捷键大全,教你成为电脑高手!!
  • #我与Java虚拟机的故事#连载07:我放弃了对JVM的进一步学习
  • $().each和$.each的区别
  • (1) caustics\
  • (k8s中)docker netty OOM问题记录
  • (ZT)出版业改革:该死的死,该生的生
  • (层次遍历)104. 二叉树的最大深度
  • (分享)一个图片添加水印的小demo的页面,可自定义样式
  • (附源码)spring boot火车票售卖系统 毕业设计 211004
  • (十)c52学习之旅-定时器实验
  • (一) springboot详细介绍
  • .halo勒索病毒解密方法|勒索病毒解决|勒索病毒恢复|数据库修复