【智能家居】6、语音控制及网络控制代码实现
一、语音控制
1、指令结构体编写

这个结构体定义了一个命令输入的模型。在这个模型中,包含以下几个部分:
- cmdName:一个长度为128的字符串,用于存储命令名称。
- dvicesName:一个长度为128的字符串,用于存储设备名称。
- cmd:一个长度为32的字符串,用于存储具体的命令。
- Init:一个函数指针,该函数接受三个参数:指向
InputCmd结构体实例的指针(可以读取所有字段)、IP地址和端口号,并返回一个整数值。 - getCmd:一个函数指针,该函数接受一个参指向
InputCmd结构体实例的指针(可以读取所有字段)并返回一个整数值。 - log:一个长度为1024的字符数组,用于存储日志信息。
- fd:一个整数,文件描述符。
- next:一个指向结构体本身的指针,用于操控链表。
imputCmd.h
#include <stdio.h>
#include <wiringPi.h>
#include <wiringSerial.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <string.h>struct InputCmd{char cmdName[128];char devicesName[128];char cmd[32];int (*Init)(struct InputCmd *cmd,char *ipAdress,char *port);int (*getCmd)(struct InputCmd *cdm);char *log[1024];int fd;char port[12];char ipAddress[32];int s_fd;struct InputCmd *next;
};
2、实例化对象
(1)结构体变量初始化

一个结构体变量初始化的例子,声明并初始化了一个`InputCmd`类型的变量`voice`,并将每个字段分别赋值。
- cmdName字段被赋值为字符串"voice"。
- devicesName字段被赋值为字符串"/dev/ttyAMA0"。
- cmd字段被赋值为空字符串。
- init字段被赋值为`voice_init`函数的地址。
- getCmd字段被赋值为`voice_getCmd`函数的地址。
- log字段被赋值为空字符串。
(2)对应字段函数编写
初始化串口,获取文件描述符

fd=serialOpen(voice->devicesName,9600)它将尝试以9600波特率打开名为 voice->devicesName 的串行端口,并将返回的文件描述符(File Descriptor)赋值给变量 fd。这里的 voice->devicesName 是指向一个字符串的指针,该字符串包含要打开的串行端口的名称或路径。例如,在Linux系统上,这可能是类似于 "/dev/ttyS0" 或 "/dev/ttyUSB0" 的设备文件路径。
9600 是波特率,它决定了数据传输的速度,表示每秒传输9600个数据位。如果 serialOpen 函数成功打开串行端口,它将返回一个非负整数。如果打开失败,函数通常会返回 -1。
读取指令

从voicer->fd指向的串行端口读取数据,并将这些数据存储在voicer->command缓冲区中。
插入链表等待被调用

Voice.c
#include "inputCmd.h"int voice_init(struct InputCmd *voice,char *ipAdress,char *port){int fd;if((fd=serialOpen(voice->devicesName,9600))==-1){exit(-1);}voice->fd=fd;return fd;
}int voice_getCmd(struct InputCmd *voice){int nread=0;nread=read(voice->fd,voice->cmd,sizeof(voice->cmd));if(nread==0){printf("usart for voice read over time\n");}elsereturn nread;
}struct InputCmd voice={.cmdName="voice",.devicesName="/dev/ttyAMA0",.cmd={'\0'},.Init=voice_init,.getCmd=voice_getCmd,.log={'\0'}
};struct InputCmd* addVoiceToInputCmdLink(struct InputCmd *phead){if(phead==NULL){return &voice;}else{voice.next=phead;phead=&voice;}
}
3、将插入链表函数加入inputCmd.h等待被调用

4、main函数编写
插入链表等待被调用

mainPro.c
#include <stdio.h>
#include <string.h>#include "controlDevices.h"
#include "inputCmd.h"struct Devices *findDevicesName(char *name,struct Devices *phead){struct Devices *tmp=phead;if(phead==NULL){return NULL;}else{while(tmp!=NULL){if(strcmp(tmp->devicesName,name)==0){return tmp;}tmp=tmp->next;}return NULL;}
}int main(){if(wiringPiSetup()==-1){return -1;}struct Devices *pdevicesHead=NULL;struct InputCmd *pinputCmdHead=NULL;pdevicesHead=addbathroomLightToDevicesLink(pdevicesHead);pdevicesHead=addupstairLightToDevicesLink(pdevicesHead);pdevicesHead=addlivingroomLightToDevicesLink(pdevicesHead);pdevicesHead=addrestaurantLightToDevicesLink(pdevicesHead);pdevicesHead=addFireToDevicesLink(pdevicesHead);pinputCmdHead=addVoiceToInputCmdLink(pinputCmdHead);char name[128]={'\0'};struct Devices *tmp=NULL;while(1){printf("INPUT:\n");scanf("%s",name);tmp=findDevicesName(name,pdevicesHead);if(tmp!=NULL){tmp->devicesInit(tmp->pinNum);tmp->open(tmp->pinNum); tmp->readStatus(tmp->pinNum);}}return 0;
}
二、网络控制
1、结构体变量初始化

2、对应字段函数编写
(1)socket_Init
1、创建一个服务器端的套接字并初始化服务器地址结构

- 定义两个socket描述符变量:
s_fd为服务端socket描述符,c_fd为客户端socket描述符。 - 定义两个
sockaddr_in结构体变量:s_addr存储服务器端网络地址信息,c_addr存储客户端网络地址信息。 - 使用
memset()函数清零这两个结构体变量,确保它们的所有成员初始化为0。 - 创建服务器端socket

调用 socket() 函数创建一个基于IPv4协议的流式(TCP)socket。如果返回值 s_fd 为-1,则表示socket创建失败,程序通过 perror("socket") 输出错误信息,并使用 exit(-1) 终止程序执行。
- 初始化服务器地址结构体:

s_addr.sin_family = AF_INET:设置地址族为Internet(IPv4)。
s_addr.sin_port = htons(atoi(socketInit->port)): 将字符串形式的端口号转换为整型数字,然后使用 htons() 函数将端口号从主机字节序转换为网络字节序。
inet_aton(socketInit->ipaddress, &s_addr.sin_addr): 将字符串形式的IP地址转换为二进制格式,并存入到s_addr.sin_addr中。这样就完成了服务器地址的配置,接下来可以调用bind()函数绑定这个地址和端口到刚创建的socket上。(需要包含头文件#include <arpa/inet.h>)
2、将创建的socket与特定的IP地址和端口号绑定

s_fd是之前通过socket()函数创建的socket描述符。(struct sockaddr *)&s_addr指向已经初始化好的服务器地址结构体s_addr。由于bind()函数的第二个参数要求指向一个通用套接字地址结构(struct sockaddr *),所以我们需要类型转换。
如果 bind() 调用成功,那么该socket就会监听指定的IP地址和端口上的连接请求。如果失败,则会返回一个错误码,通常需要通过 perror() 或检查 errno 来获取错误信息并进行相应的处理。
3、 将socket设置为监听模式

s_fd是之前创建并绑定到特定IP地址和端口的socket描述符。- 第二个参数(这里是10)表示backlog,即在服务器开始拒绝连接请求前,可以排队等待处理的最大连接数。当这个数目达到后,新的连接请求将会被拒绝,直到服务器完成对已有连接的处理。

将已经初始化并设置为监听状态的socket描述符 s_fd 赋值给结构体 socketInit 的成员变量 s_fd
(2) socket_getCmd
1、声明变量与初始化c_addr结构体

int c_fd;:声明一个整型变量 c_fd,用于存储接受到的客户端socket描述符。
int n_read = 0;:声明一个整型变量 n_read,用于记录从客户端socket读取数据的数量。
struct sockaddr_in c_addr;:定义一个结构体变量 c_addr,用于存储客户端的网络地址信息。
memset(&c_addr, 0, sizeof(struct sockaddr_in));:清零 c_addr 结构体,确保所有成员初始化为0。
int len = sizeof(struct sockaddr_in);:声明并初始化一个整型变量 len,表示要获取的客户端地址结构体的大小。
2、等待并接受来自客户端的连接请求

c_fd = accept(socketGet.s_fd, (struct sockaddr *)&c_addr, &len);:使用 accept() 函数等待并接受来自客户端的连接请求。
- 第一个参数是服务端监听的socket描述符(即之前创建并调用
listen()的s_fd)。 - 第二个参数是一个指向
sockaddr_in结构体的指针,用于接收已连接客户端的地址信息。 - 第三个参数是一个指向整数的指针,用于接收实际填充到地址结构体中的字节数。
3、从已连接的客户端socket (c_fd) 读取数据

n_read = read(c_fd, socketGet->command, sizeof(socket_command));:使用 read() 函数从已连接的客户端socket (c_fd) 读取数据,并将数据存入 socketGet 结构体中 command 成员指定的缓冲区。第二个参数是缓冲区的起始地址,第三个参数是期望读取的最大字节数。

对于 read() 函数的返回值进行判断:
- 如果
n_read == -1,说明读取过程中发生错误,通过perror("read")输出错误信息。 - 如果
n_read > 0,则打印出读取到的数据长度。 - 如果
n_read == 0,通常意味着客户端已经关闭了连接,此时输出 "client quit!",然后可能执行相应的退出循环操作(这里没有显示完整的上下文,所以无法确定是否有break语句所在的循环)。

最后返回 n_read,代表本次从客户端读取数据的实际字节数。
(3)插入链表等待被调用
socket.c
#include "inputCmd.h"int socket_init(struct InputCmd *socketInit,char *ipAdress,char *port){int s_fd,c_fd;//s_fd:服务(service)端socket描述符;c_fd:客户(client)端socket描述符struct sockaddr_in s_addr;//s_addr:存储服务器端网络地址信息struct sockaddr_in c_addr;//c_addr:存储客户端网络地址信息memset(&s_addr,0,sizeof(struct sockaddr_in));//初始化结构体memset(&c_addr,0,sizeof(struct sockaddr_in));//sockets_fd=socket(AF_INET,SOCK_STREAM,0);if(s_fd==-1){perror("socket");exit(-1);}s_addr.sin_family=AF_INET;s_addr.sin_port=htons(atoi(socketInit->port));inet_aton(socketInit->ipAddress,&s_addr.sin_addr);//bindbind(s_fd,(struct sockaddr *)&s_addr,sizeof(struct sockaddr_in));//listenlisten(s_fd,10);socketInit->s_fd=s_fd;return s_fd;
}int socket_getCmd(struct InputCmd *Socket){int c_fd;int n_read=0;struct sockaddr_in c_addr;memset(&c_addr,0,sizeof(struct sockaddr_in));int len=sizeof(struct sockaddr_in);c_fd=accept(Socket->s_fd,(struct sockaddr *)&c_addr,&len);n_read=read(c_fd,Socket->cmd,sizeof(Socket->cmd));if(n_read==-1){perror("read");}else if(n_read>0){printf("\n");printf("get:%d\n",n_read);}else{printf("client quit!\n");}return n_read;
}struct InputCmd Socket={.cmdName="socket",.cmd={'\0'},.Init=socket_init,.getCmd=socket_getCmd,.log={'\0'},.port="8080",.ipAddress="172.20.10.9"
};struct InputCmd* addSocketToInputCmdLink(struct InputCmd *phead){if(phead==NULL){return &Socket;}else{Socket.next=phead;phead=&Socket;}
}
3、将插入链表函数加入inputCmd.h等待被调用
#include <stdio.h>
#include <wiringPi.h>
#include <wiringSerial.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <string.h>struct InputCmd{char cmdName[128];char devicesName[128];char cmd[32];int (*Init)(struct InputCmd *cmd,char *ipAdress,char *port);int (*getCmd)(struct InputCmd *cdm);char *log[1024];int fd;char port[12];char ipAddress[32];int s_fd;struct InputCmd *next;
};struct InputCmd* addVoiceToInputCmdLink(struct InputCmd *phead);
struct InputCmd* addSocketToInputCmdLink(struct InputCmd *phead);
4、main函数编写

mainPro.c
#include <stdio.h>
#include <string.h>#include "controlDevices.h"
#include "inputCmd.h"struct Devices *findDevicesName(char *name,struct Devices *phead){struct Devices *tmp=phead;if(phead==NULL){return NULL;}else{while(tmp!=NULL){if(strcmp(tmp->devicesName,name)==0){return tmp;}tmp=tmp->next;}return NULL;}
}int main(){if(wiringPiSetup()==-1){return -1;}struct Devices *pdevicesHead=NULL;struct InputCmd *pinputCmdHead=NULL;pdevicesHead=addbathroomLightToDevicesLink(pdevicesHead);pdevicesHead=addupstairLightToDevicesLink(pdevicesHead);pdevicesHead=addlivingroomLightToDevicesLink(pdevicesHead);pdevicesHead=addrestaurantLightToDevicesLink(pdevicesHead);pdevicesHead=addFireToDevicesLink(pdevicesHead);pinputCmdHead=addVoiceToInputCmdLink(pinputCmdHead);pinputCmdHead=addSocketToInputCmdLink(pinputCmdHead);char name[128]={'\0'};struct Devices *tmp=NULL;while(1){printf("INPUT:\n");scanf("%s",name);tmp=findDevicesName(name,pdevicesHead);if(tmp!=NULL){tmp->devicesInit(tmp->pinNum);tmp->open(tmp->pinNum); tmp->readStatus(tmp->pinNum);}}return 0;
}
三、测试
将工程放入树莓派中,编译通过,无错误无警告

四、当前阶段源代码
链接:https://pan.baidu.com/s/1vh8smf725jrxSUlDOHFOfg
提取码:sail
相关文章:
【智能家居】6、语音控制及网络控制代码实现
一、语音控制 1、指令结构体编写 这个结构体定义了一个命令输入的模型。在这个模型中,包含以下几个部分: cmdName:一个长度为128的字符串,用于存储命令名称。dvicesName:一个长度为128的字符串,用于存储设…...
Windows断开映射磁盘提示“此网络连接不存在”,并且该磁盘直在资源管理器中
1、打开注册表编辑器 快捷键winR 打开“运行”, 输入 regedit 2、 删除下列注册表中和无法移除的磁盘相关的选项 \HKEY_CURRENT_USER\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\MountPoints2\ 3、打开“任务管理器”,重新启动“Windows资源…...
Java项目要不要部署在Docker里?
部署Java项目有很多种方式,传统的方式是直接在物理机或虚拟机上部署应用,但为什么现在容器化部署变得越来越流行, 个人觉得原因有以下几个: 1、 环境一致性:使用Docker可以确保开发、测试和生产环境的一致性ÿ…...
Linux 入门基础知识(一)—— Linux的基本使用
Linux 入门基础知识 一、Linux的基本使用和配置1.1、终端1.2、消耗内存1.3、运行级别1.6、登录前欢迎语1.5、登录后欢迎语1.6、shell1.7、ps aux1.8、设置主机名1.9、whoami和who am i1.10、命令提示符 二、Linux执行命令的过程详解和命令类型2.1、命令执行2.2、hash缓存表2.3、…...
纯静态微信小程序水果商城
首页代码 <view class"container"><!-- 轮播图 --><view class"swiper-container"><swiper class"screen-swiper" indicator-dots"true" circular"true" autoplay"true" interval"300…...
操作系统-线程的实现方式和多线程模型(用户级线程 内核级线程 多线程模型的情况)和线程的状态,转换,组织,控制
文章目录 线程的实现方式和多线程模型总览线程的实现方式用户级线程内核级线程多线程模型一对一多对一多对多 小结 线程的状态,转换,组织,控制总览 线程的状态与转换线程的组织与控制 线程的实现方式和多线程模型 总览 线程的实现方式 用户级线程 程序自己通过自己设计的线程…...
计算机网络 应用层
文章目录 应用层域名系统 DNS域名系统概述互联网的域名结构域名服务器 文件传送协议FTP 概述FTP 的基本工作原理简单文件传送协议 TFTP 远程终端协议 TELNET万维网 WWW统一资源定位符 URL超文本传送协议 HTTP万维网的信息检索系统 电子邮件电子邮件概述简单邮件传送协议 SMTP邮…...
拥有你的微调大模型,零代码定制+算力免费送
2023年,ChatGPT的横空出世掀起了人工智能领域的新一轮革命。 很快大模型的潮水就越过广袤的太平洋,席卷国内科技圈和投资圈,几乎所有科技公司都在押注大模型。根据工信部赛迪研究院数据,预计2023年我国语言大模型市场规模将达到1…...
《Docker技术革命:从虚拟机到容器化,全面解析Docker的原理与应用-上篇》
文章目录 Docker为什么会出现总结 Docker的思想Docker历史总结 Docker能干嘛虚拟机技术虚拟机技术的缺点 容器化技术Docker和虚拟机技术的区别 Docker概念Docker的基本组成镜像(image)容器(container)仓科(repository)…...
C,C++,C# 的区别
C,C,C# 是三种不同的编程语言。它们有一些共同点,但也有很多区别。 C是一种过程式编程语言,被广泛用于系统级编程和嵌入式开发。它具有简单的语法和较小的标准库,适合于高效的低级编程和处理底层细节。 C是一种面向对…...
操作筛选器的 1 个应用实例:自动启用事务
前言 在数据库操作过程中,有一个概念是绕不开的,那就是事务。 事务能够确保一系列数据库操作要么全部成功提交,要么全部失败回滚,保证数据的一致性和完整性。 在 Asp.Net Core Web API 中,我们可以使用操作筛选器给…...
搭建基于Java的分布式爬虫系统
目录 前言 一、分布式爬虫系统的架构设计 二、系统搭建步骤 1. 创建爬虫项目 2. 导入相关依赖 3. 编写分布式爬虫系统的核心代码 3.1 节点管理器(Node Manager) 3.2 调度器(Scheduler) 3.3 下载器(Downloader…...
rancher证书过期问题处理
问题 起初,打开rancher ui页面打不开,telnet rancher的服务端口也不通。查看rancher 控制节点,日志显示,X509:certificate has expired or is not ye valid。证书已过期 解决 现在网上大部分的解决方案都是针对的2…...
Spring Boot 中文件上传
Spring Boot 中文件上传 一、MultipartFile二、单文件上传案例三、多文件上传案例四、Servlet 规范五、Servlet 规范实现文件上传 上传文件大家用的最多的就是 Apache Commons FileUpload,这个库使用非常广泛。Spring Boot3 版本中已经不能使用了。代替它的是 Sprin…...
2023年06月CCF-GESP编程能力等级认证Python编程一级真题解析
一、单选题(共15题,共30分) 第1题 以下不属于计算机输出设备的有()。 A:麦克风 B:音箱 C:打印机 D:显示器 答案:A 第2题 ChatGPT 是 OpenAI 研发的聊天机器人程序,它能通过理解和学习人类的语言来进行对话,还能根据聊天的上下文进行互动,完成很多工作。请你…...
unity 使用数字图片来代替数字0到9显示
using System.Collections; using System.Collections.Generic; using UnityEngine; using UnityEngine.UI; public class Trackracecomplete : MonoBehaviour { /// /// 数字图片 /// [SerializeField] private Sprite[] sprites; private string _Time “23:57:49”; [Ser…...
单片机如何实现延时1ms或者1us
1us //适配主频为120MHz的单片机 void Delay_us(int16_t nus) {int32_t temp; SysTick->LOAD nus*15; //120MHzSysTick->VAL0X00;SysTick->CTRL0X01;do { tempSysTick->CTRL;}while((temp&0x01)&&(!(temp&(1<<16))));SysTick->CTRL0x0…...
全国网络安全行业职业技能大赛WP
word_sercet 文档被加密 查看图片的属性 在备注可以看到解压密码 解密成功 在选项里面把隐藏的文本显示出来 可以看到ffag easy_encode 得到一个bmp二维码 使用qr research 得到的密文直接放瑞士军刀 base32解码base64解码hex解码 dir_pcap 直接搜索flag 发现flag…...
【Python函数与模块】(7)模块的分类与好处
文章目录 一、模块分类二、模块的好处 一、模块分类 Python标准模块(内置模块,标准库) 第三方模块/库(pypi.org) 自定义模块 二、模块的好处 可维护性更强 方便代码重用...
如何安全地多开Facebook/Twitter/TK/Ins等账号?
随着社交媒体的普及,人们需要在不同平台上管理多个账号。然而,如何安全地多开这些账号却是一个需要关注的问题。本文将介绍如何安全地多开Facebook、twitter、YouTube、TikTok等平台账号的方法。 重要关联因素: 1. 隐私和安全:保…...
vscode里如何用git
打开vs终端执行如下: 1 初始化 Git 仓库(如果尚未初始化) git init 2 添加文件到 Git 仓库 git add . 3 使用 git commit 命令来提交你的更改。确保在提交时加上一个有用的消息。 git commit -m "备注信息" 4 …...
Redis相关知识总结(缓存雪崩,缓存穿透,缓存击穿,Redis实现分布式锁,如何保持数据库和缓存一致)
文章目录 1.什么是Redis?2.为什么要使用redis作为mysql的缓存?3.什么是缓存雪崩、缓存穿透、缓存击穿?3.1缓存雪崩3.1.1 大量缓存同时过期3.1.2 Redis宕机 3.2 缓存击穿3.3 缓存穿透3.4 总结 4. 数据库和缓存如何保持一致性5. Redis实现分布式…...
MFC内存泄露
1、泄露代码示例 void X::SetApplicationBtn() {CMFCRibbonApplicationButton* pBtn GetApplicationButton();// 获取 Ribbon Bar 指针// 创建自定义按钮CCustomRibbonAppButton* pCustomButton new CCustomRibbonAppButton();pCustomButton->SetImage(IDB_BITMAP_Jdp26)…...
PPT|230页| 制造集团企业供应链端到端的数字化解决方案:从需求到结算的全链路业务闭环构建
制造业采购供应链管理是企业运营的核心环节,供应链协同管理在供应链上下游企业之间建立紧密的合作关系,通过信息共享、资源整合、业务协同等方式,实现供应链的全面管理和优化,提高供应链的效率和透明度,降低供应链的成…...
鸿蒙中用HarmonyOS SDK应用服务 HarmonyOS5开发一个医院挂号小程序
一、开发准备 环境搭建: 安装DevEco Studio 3.0或更高版本配置HarmonyOS SDK申请开发者账号 项目创建: File > New > Create Project > Application (选择"Empty Ability") 二、核心功能实现 1. 医院科室展示 /…...
2025盘古石杯决赛【手机取证】
前言 第三届盘古石杯国际电子数据取证大赛决赛 最后一题没有解出来,实在找不到,希望有大佬教一下我。 还有就会议时间,我感觉不是图片时间,因为在电脑看到是其他时间用老会议系统开的会。 手机取证 1、分析鸿蒙手机检材&#x…...
实现弹窗随键盘上移居中
实现弹窗随键盘上移的核心思路 在Android中,可以通过监听键盘的显示和隐藏事件,动态调整弹窗的位置。关键点在于获取键盘高度,并计算剩余屏幕空间以重新定位弹窗。 // 在Activity或Fragment中设置键盘监听 val rootView findViewById<V…...
Typeerror: cannot read properties of undefined (reading ‘XXX‘)
最近需要在离线机器上运行软件,所以得把软件用docker打包起来,大部分功能都没问题,出了一个奇怪的事情。同样的代码,在本机上用vscode可以运行起来,但是打包之后在docker里出现了问题。使用的是dialog组件,…...
AI病理诊断七剑下天山,医疗未来触手可及
一、病理诊断困局:刀尖上的医学艺术 1.1 金标准背后的隐痛 病理诊断被誉为"诊断的诊断",医生需通过显微镜观察组织切片,在细胞迷宫中捕捉癌变信号。某省病理质控报告显示,基层医院误诊率达12%-15%,专家会诊…...
MySQL 8.0 事务全面讲解
以下是一个结合两次回答的 MySQL 8.0 事务全面讲解,涵盖了事务的核心概念、操作示例、失败回滚、隔离级别、事务性 DDL 和 XA 事务等内容,并修正了查看隔离级别的命令。 MySQL 8.0 事务全面讲解 一、事务的核心概念(ACID) 事务是…...

