蓝牙模块HC-08+WIFI模块ESP-01S

蓝牙模块

又叫蓝牙串口模块。
串口透传技术:透传即透明传送,是指在数据的传输过程中,通过无线的方式使这组数据不发生任何形式的改变,仿佛传输过程是透明的一样,同时保证传输的质量,原封不动地道了最终接收者手里。
下载hc-08板子上二维码对应的HC蓝牙助手app,将hc08接入单片机(注意tx,rx反接),此时烧录以下代码:

#include "reg52.h"
#include <intrins.h>
#include <string.h>#define SIZE 12
sfr AUXR = 0x8E;
sbit led = P3^7;char cmd[12];void Delay1000ms()		//@11.0592MHz
{unsigned char i, j, k;_nop_();i = 8;j = 1;k = 243;do{do{while (--k);} while (--j);} while (--i);
}void UartInit(void)		//9600bps@11.0592MHz
{SCON = 0x50;	//定义串口工作方式为方式1,8位UART//PCON初值为00x1 0000符合我们的要求,我们只需要SMOD=0不加倍即可AUXR = 0x01;TMOD &= 0x0F;TMOD |= 0x20;	//配置定时器1为8为自动重载定时器TL1 = 0xFD;TH1 = 0xFD;	//9600波特率初值TR1 = 1;	//打开中断EA = 1;	//开启总中断ES = 1;	//开启UART中断
}void sendByte(char data_msg){SBUF = data_msg;//当8位数据没有传送结束时,卡在while循环中while(!TI);	//TI发送中断请求标志位,第8位数据传送结束时,硬件自动置1TI = 0;
}void sendString(char* str){while(*str != '\0'){sendByte(*str++);}
}void main()
{led = 1;//配置C51串口的通信方式UartInit();while(1){Delay1000ms();//往发送缓冲区写入数据,就完成数据的发送sendString("hello world\r\n");}
}void Uart_Handler() interrupt 4
{static int i = 0;	//if(RI){	//中断处理函数中,对于接收中断的响应RI = 0;cmd[i] = SBUF;i++;if(i == SIZE) i=0;if(strstr(cmd,"en")){	//判断cmd中是否有“en”子串led = 0;i=0;memset(cmd,'\0',SIZE);}if(strstr(cmd,"se")){led = 1;i=0;memset(cmd,'\0',SIZE);}}if(TI);
}

打开手机app即可每隔一秒接收到一个hello world同时可以通过手机发送open或close开/关灯.
同时可以通过AT指令进行快乐的玩耍

WIFI模块

初始配置和验证

为了将来与单片机的波特率保持一致,将波特率修改为9600

AT+UART=9600,8,1,0,0

入网设置

1.设置工作模式

AT+CWMODE=3	//1.是station(设备)模式2.是AP(路由)模式3.双模OK

2.以设备模式接入家中的路由器

AT+CWJAP="wifipig","66668888"
WIFI CONNECTED
WIFI GOT IPOK

3.查询IP地址

AT+CIFSR
+CIFSR:APIP,"192.168.4.1"
+CIFSR:APMAC,"a6:cf:12:c1:ec:eb"
+CIFSR:STAIP,"192.168.31.46"
+CIFSR:STAMAC,"a4:cf:12:c1:ec:eb"OK

4.连接服务器

AT+CIPSTART="TCP","192.168.31.152",8880
CONNECTOK

5.发送数据

AT+CIPSEND=4	//设置即将发送的数据长度  4个字节
> 				//注意在发送数据的时候要取消勾选发送新行,在上传指令的时候要勾选发送新行
Recv 4 bytesSEND OK

透传发送数据

基于上述连接服务器之后

AT+CIPMODE=1	//开启透传模式OKAT+CIPSEND>				//即可肆意传输 当你不想传输了发送一个“+++”即可退出透传模式

将wifi模块接入单片机进行开发

首先我们将以下代码烧录至单片机中:

#include "reg52.h"
#include <intrins.h>
#include <string.h>#define SIZE 12
sfr AUXR = 0x8E;
sbit led = P3^7;char cmd[SIZE];code char LJWL[]  = "AT+CWJAP=\"wifipi\",\"66668888\"\r\n";
code char LJFWQ[] = "AT+CIPSTART=\"TCP\",\"192.168.31.152\",8880\r\n";
char TCMS[]  = "AT+CIPMODE=1\r\n";
char SJCS[]  = "AT+CIPSEND\r\n";void Delay1000ms()		//@11.0592MHz
{unsigned char i, j, k;_nop_();i = 8;j = 1;k = 243;do{do{while (--k);} while (--j);} while (--i);
}void UartInit(void)		//9600bps@11.0592MHz
{SCON = 0x50;	//定义串口工作方式为方式1,8位UART//PCON初值为00x1 0000符合我们的要求,我们只需要SMOD=0不加倍即可AUXR = 0x01;TMOD &= 0x0F;TMOD |= 0x20;	//配置定时器1为8为自动重载定时器TL1 = 0xFD;TH1 = 0xFD;	//9600波特率初值TR1 = 1;	//打开中断EA = 1;	//开启总中断ES = 1;	//开启UART中断
}void sendByte(char data_msg){SBUF = data_msg;//当8位数据没有传送结束时,卡在while循环中while(!TI);	//TI发送中断请求标志位,第8位数据传送结束时,硬件自动置1TI = 0;
}void sendString(char* str){while(*str != '\0'){sendByte(*str++);}
}void main()
{led = 1;//配置C51串口的通信方式UartInit();while(1){//Delay1000ms();//往发送缓冲区写入数据,就完成数据的发送//sendString("hello world\r\n");sendString(LJWL);Delay1000ms();Delay1000ms();Delay1000ms();Delay1000ms();Delay1000ms();sendString(LJFWQ);Delay1000ms();Delay1000ms();Delay1000ms();Delay1000ms();Delay1000ms();sendString(TCMS);Delay1000ms();Delay1000ms();Delay1000ms();Delay1000ms();Delay1000ms();sendString(SJCS);Delay1000ms();Delay1000ms();Delay1000ms();Delay1000ms();Delay1000ms();}
}void Uart_Handler() interrupt 4
{static int i = 0;	//if(RI){	//中断处理函数中,对于接收中断的响应RI = 0;cmd[i] = SBUF;i++;if(i == SIZE) i=0;if(strstr(cmd,"en")){	//判断cmd中是否有“en”子串led = 0;i=0;memset(cmd,'\0',SIZE);}if(strstr(cmd,"se")){led = 1;i=0;memset(cmd,'\0',SIZE);}}if(TI);
}

依次传输连接网络、连接服务器、透传模式、数据传输的指令,先传到电脑上检查一个格式是否正确。
之后可以将wifi模块接入单片机,依次将上述指令传输给wifi模块,但是我们如何知道wifi模块是否正确接收到指令并返回正确的回复?
白盒测试
我们可以通过单片机传输指令至8266wifi模块,之后将8266的tx与pc端串口助手的rx连接,查看8266返回的数据以检查是否正确接收指令并作出正确的回复。
在这里插入图片描述
可以看到8266模块确实正确接收到了指令,并且成功连接网络、连接服务器、开启透传模式进行了数据传输,因此代码正确。
此时我们通过网络向8266发送数据后,由于8266的tx口连着pc端因此会在pc端的串口助手中接收到数据,当我们将8266的tx口与单片机相连,就可以实现通过网络来操控单片机的效果。

通过TCP通信控制LED灯

将8266与单片机相连,修改串口中断中的代码:

#include "reg52.h"
#include <intrins.h>
#include <string.h>#define SIZE 12
sfr AUXR = 0x8E;
sbit led = P3^7;char cmd[SIZE];code char LJWL[]  = "AT+CWJAP=\"wifipi\",\"66668888\"\r\n";
code char LJFWQ[] = "AT+CIPSTART=\"TCP\",\"192.168.31.152\",8880\r\n";
char TCMS[]  = "AT+CIPMODE=1\r\n";
char SJCS[]  = "AT+CIPSEND\r\n";void Delay1000ms()		//@11.0592MHz
{unsigned char i, j, k;_nop_();i = 8;j = 1;k = 243;do{do{while (--k);} while (--j);} while (--i);
}void UartInit(void)		//9600bps@11.0592MHz
{SCON = 0x50;	//定义串口工作方式为方式1,8位UART//PCON初值为00x1 0000符合我们的要求,我们只需要SMOD=0不加倍即可AUXR = 0x01;TMOD &= 0x0F;TMOD |= 0x20;	//配置定时器1为8为自动重载定时器TL1 = 0xFD;TH1 = 0xFD;	//9600波特率初值TR1 = 1;	//打开中断EA = 1;	//开启总中断ES = 1;	//开启UART中断
}void sendByte(char data_msg){SBUF = data_msg;//当8位数据没有传送结束时,卡在while循环中while(!TI);	//TI发送中断请求标志位,第8位数据传送结束时,硬件自动置1TI = 0;
}void sendString(char* str){while(*str != '\0'){sendByte(*str++);}
}void main()
{int mark = 0;led = 1;//配置C51串口的通信方式UartInit();while(1){//Delay1000ms();//往发送缓冲区写入数据,就完成数据的发送//sendString("hello world\r\n");if(mark == 0){sendString(LJWL);Delay1000ms();Delay1000ms();Delay1000ms();Delay1000ms();Delay1000ms();sendString(LJFWQ);Delay1000ms();Delay1000ms();Delay1000ms();Delay1000ms();Delay1000ms();sendString(TCMS);Delay1000ms();Delay1000ms();Delay1000ms();Delay1000ms();Delay1000ms();sendString(SJCS);Delay1000ms();Delay1000ms();Delay1000ms();Delay1000ms();Delay1000ms();mark = 1;}else{sendString("hello world\r\n");Delay1000ms();}}
}void Uart_Handler() interrupt 4
{if(RI){	//中断处理函数中,对于接收中断的响应RI = 0;cmd[0] = SBUF;if(cmd[0] == '1'){led = 0;}if(cmd[0] == '0'){led = 1;}}if(TI);
}

但是这段代码在运行时,我们无法看到任何连接的信息,不知道连接是否正确进行,因此代码是不完善的,在有时候我们无法连接上wifi因此我们还是需要借助上述白盒测试的方法来检查返回值,程序可能会因为在5s内连接不上wifi而导致后续一系列操作都发生错误,因此我们需要优化我们的代码:
基本上当我能够正确接入wifi后续的操作就会成功,而接入wifi后会返回的提示为WIFI CONNECTED
WIFI GOT IP
在这里插入图片描述
根据程序正确运行时的返回值来进行判断.
这里记录一下在编程过程中遇到的一个错误卡了我好久

void Uart_Handler() interrupt 4
{static int i = 0;char tmp;if(RI){	//中断处理函数中,对于接收中断的响应RI = 0;tmp = SBUF;if(tmp == 'W' || tmp == 'O' || tmp == 'L'){i = 0;}buffer[i] = tmp;i++;if(buffer[0] == 'W' && buffer[5] == 'G'){	//当显示WIFI GOR IP时表示网络已经接入,修改标志位AT_Connect_Net_Flag = 1;memset(buffer,'\0',SIZE);}if(buffer[0] == 'O' && buffer[1] == 'K'){AT_OK_Flag = 1;memset(buffer,'\0',SIZE);}if(buffer[0] == 'L' && buffer[2] == '1'){D5 = 0;memset(buffer,'\0',SIZE);}if(buffer[0] == 'L' && buffer[2] == '0'){D6 = 1;memset(buffer,'\0',SIZE);}if(i == 12) i=0;memset(buffer,'\0',SIZE);//出错了!!!!}if(TI);
}

在调试过程中,我用之前的白盒测试方法来进行检验,但是当我给单片机发送OK信号时,没有用,卡了很久,最后发现是,倒数第四行多写了一个memset。这段代码的含义是,每次写入一个字节后,会对每个写入的字符进行判断,如果是W、O、L开头,会强制将下标变为0,为了能够方便下面的判断,因为当我们连入wifi后,8266模块会返回WIFI GOT IP,第一个字符为W,第6个字符为G,因此我们可以判断buffer[0]是否为W,buffer[5]是否为G从而判断是否成功入网,如果满足条件,将入网标志位置一,同时清空buffer数组。但是上述代码会在我每次读取完一个字符后,将buffer清空,因此永远也无法达到效果。
修改完毕后,就可以达到我们需要的效果了:
在这里插入图片描述
检验完毕后,我们直接连接到单片机即可。

#include "reg52.h"
#include <intrins.h>
#include <string.h>#define SIZE 12
sfr AUXR = 0x8E;
sbit D5 = P3^7;
sbit D6 = P3^6;char buffer[SIZE];code char LJWL[]  = "AT+CWJAP=\"wifipig\",\"66668888\"\r\n";
code char LJFWQ[] = "AT+CIPSTART=\"TCP\",\"192.168.31.152\",8880\r\n";
char RESET[] = "AT+RST\r\n";
char TCMS[]  = "AT+CIPMODE=1\r\n";
char SJCS[]  = "AT+CIPSEND\r\n";char AT_Connect_Net_Flag = 0;
char AT_OK_Flag = 0;void Delay1000ms()		//@11.0592MHz
{unsigned char i, j, k;_nop_();i = 8;j = 1;k = 243;do{do{while (--k);} while (--j);} while (--i);
}void UartInit(void)		//9600bps@11.0592MHz
{SCON = 0x50;	//定义串口工作方式为方式1,8位UART//PCON初值为00x1 0000符合我们的要求,我们只需要SMOD=0不加倍即可AUXR = 0x01;TMOD &= 0x0F;TMOD |= 0x20;	//配置定时器1为8为自动重载定时器TL1 = 0xFD;TH1 = 0xFD;	//9600波特率初值TR1 = 1;	//打开中断EA = 1;	//开启总中断ES = 1;	//开启UART中断
}void sendByte(char data_msg){SBUF = data_msg;//当8位数据没有传送结束时,卡在while循环中while(!TI);	//TI发送中断请求标志位,第8位数据传送结束时,硬件自动置1TI = 0;
}void sendString(char* str){while(*str != '\0'){sendByte(*str++);}
}void main()
{int mark = 0;D5 = D6 = 1;//配置C51串口的通信方式UartInit();Delay1000ms();	//给wifi模块上电时间sendString(LJWL);while(!AT_Connect_Net_Flag);while(!AT_OK_Flag);AT_OK_Flag = 0;sendString(LJFWQ);while(!AT_OK_Flag);AT_OK_Flag = 0;sendString(TCMS);while(!AT_OK_Flag);AT_OK_Flag = 0;sendString(SJCS);while(!AT_OK_Flag);if(AT_Connect_Net_Flag){D5 = 0;	//点亮第一盏灯说明入网成功}if(AT_OK_Flag){D6 = 0;	//点亮第二盏led说明网络连接成功并成功打开透传模式}while(1){Delay1000ms();sendString("hello world\r\n");}	
}void Uart_Handler() interrupt 4
{static int i = 0;char tmp;if(RI){	//中断处理函数中,对于接收中断的响应RI = 0;tmp = SBUF;if(tmp == 'W' || tmp == 'O' || tmp == 'L' || tmp == "F"){i = 0;}buffer[i] = tmp;i++;if(buffer[0] == 'W' && buffer[5] == 'G'){	//当显示WIFI GOR IP时表示网络已经接入,修改标志位AT_Connect_Net_Flag = 1;memset(buffer,'\0',SIZE);}if(buffer[0] == 'O' && buffer[1] == 'K'){AT_OK_Flag = 1;memset(buffer,'\0',SIZE);}if(buffer[0] == 'F' && buffer[1] == 'A'){for(i=0;i<5;i++){D5 = 0;Delay1000ms();D5 = 1;Delay1000ms();}//sendString(RESET);memset(buffer, '\0', SIZE);}if(buffer[0] == 'L' && buffer[2] == '1'){D5 = 0;memset(buffer,'\0',SIZE);}if(buffer[0] == 'L' && buffer[2] == '0'){D5 = 1;memset(buffer,'\0',SIZE);}if(i == 12) i=0;}//if(TI);
}

当成功连入后,两盏灯都会点亮,可以用过L-1点亮D5,L-0熄灭D5。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Qt下使用7Z源码进行压缩和解压缩

7Z压缩是一款常用的压缩算法和工具&#xff0c;本文主要介绍一款在qt环境下进行编译的压缩方法。 本人测试是可以正常跑通的&#xff0c;具体代码部分请下载&#xff1a;下载链接&#xff0c;提取码&#xff1a;ev9t 7z源码网址&#xff1a;7-Zip 7z简介&#xff1a; 7z 是…

.[[MyFile@waifu.club]].svh勒索病毒数据库恢复方案

.[[MyFilewaifu.club]].svh勒索病毒有什么特点&#xff1f; .[[MyFilewaifu.club]].svh是一种最近多发的勒索病毒&#xff0c;它通过加密受害者的文件并要求支付赎金来解锁&#xff0c;从而达到勒索钱财的目的。恢复重要数据请添加技术服务号(safe130)。以下是关于这种病毒的详…

【计算机科学速成课】笔记一

文章目录 写在前面1.计算机的早期历史2.电子计算机3.布尔运算和逻辑门4.二进制5.算术逻辑单元-ALU6.寄存器和内存 写在前面 所有的一切源于这样一个网站——CS自学指南。 这是新手小白入门计算机科学必要了解的知识——【计算机科学速成课】[40集全/精校] - Crash Course Comp…

Redis(Redis配置和订阅发布)

文章目录 1.Redis配置1.网络配置1.配置文件位置 /etc/redis.conf2.bind&#xff08;注销支持远程访问&#xff09;1.默认情况bind 127.0.0.1 只能接受本机的访问2.首先编辑配置文件3.进入命令模式输入/bind定位&#xff0c;输入n查找下一个&#xff0c;shift n查找上一个&…

恒创科技「5月活动」中国香港/美国服务器配置及价格汇总

值此 5 月&#xff0c;各大云服务商的促销活动也是接连不断。近日&#xff0c;恒创科技上线了 5 月出海乐购“惠”活动&#xff0c;从 4 月 28 日开始&#xff0c;持续整个 5 月份。活动分精选云产品、主流物理服务器、DDoS 防护、免费试用等四大专区&#xff0c;包含几十款企业…

一键生成AI数字人短视频工具推荐!

数字人是什么&#xff1f;是利用人工智能技术实现与真人直播形象的1:1克隆&#xff0c;即克隆出一个数字化的你自己&#xff0c;包括你的形象、表情、动作和声音都会被克隆下来&#xff0c;让你能够拥有接近真人的表现力。 怎样使用数字人一天生成上百条短视频&#xff0c;无需…

js浏览器请求,post请求中的参数形式和form-data提交数据时数据格式问题(2024-05-06)

浏览器几种常见的post请求方式 Content-Type 属性规定在发送到服务器之前应该如何对表单数据进行编码。 默认表单数据会编码为 "application/x-www-form-urlencoded" post请求的参数一般放在Body里。 Content-Type&#xff08;内容类型&#xff09;&#xff0c;一般…

Elasticsearch:理解人工智能相似性搜索

理解相似性搜索&#xff08;也称为语义搜索&#xff09;的指南&#xff0c;这是人工智能最新阶段的关键发现之一。 最新阶段人工智能的关键发现之一是根据相似性搜索和查找文档的能力。相似性搜索是一种比较信息的方法&#xff0c;其基于含义而非关键字。 相似性搜索也被称为语…

5月5日智渍洁对江苏某公司冷却塔清洗-智渍洁

简报&#xff1a;5月5日智渍洁对江苏某公司冷却塔清洗 5月5日智渍洁对江苏某公司冷却塔清洗 - 重庆智渍洁环保科技有限公司简报&#xff1a;5月5日智渍洁对江苏某公司冷却塔清洗https://www.zhizijie.com/hl/zixun/gongsi/236.html

linux - 主次设备号自动申请

alloc_chrdev_region 原型如下&#xff0c;该函数向内核申请一个空闲的主设备号。 alloc_chrdev_region(&g_aputriger_dev, 0, APUTRIGER_MAX_NUM, "aputriger0"); 第四个参数是我们使用cat /proc/devices 看到的名称 /*** alloc_chrdev_region() - register a…

JAVA面向对象高级部分

内部类 内部类的四种形式 内部类概述、成员内部类 代码示例 创建对象的格式 通过对象名访问内部类方法 若内外部类的成员变量名冲突&#xff0c;如何在内部类分别访问外部成员变量。 总结 静态内部类 代码示例 访问静态内部类的方法 不能在静态内部类中访问实例成员变量 …

PE文件(四)FileBuffer-ImageBuffer作业

C语言实现如下功能 2.编写一个函数&#xff0c;将RVA的值转换成FOA 将文件加载到内存时&#xff0c;已知一个数据在内存中的地址&#xff0c;将此地址转化成文件在硬盘上时的相对于文件起始地址的文件偏移地址。即将虚拟内存偏移地址转换成文件偏移地址。 说明&#xff1a;这里…

并发容器(Map、List、Set)实战及其原理

Java的集合容器框架中&#xff0c;主要有四大类别&#xff1a;List、Set、Queue、Map&#xff0c;大家熟知的这些集合类ArrayList、LinkedList、HashMap这些容器都是非线程安全的。所以&#xff0c;Java先提供了同步容器供用户使用。同步容器可以简单地理解为通过synchronized来…

Selenium自动化测试面试题全家桶

&#x1f525; 交流讨论&#xff1a;欢迎加入我们一起学习&#xff01; &#x1f525; 资源分享&#xff1a;耗时200小时精选的「软件测试」资料包 &#x1f525; 教程推荐&#xff1a;火遍全网的《软件测试》教程 &#x1f4e2;欢迎点赞 &#x1f44d; 收藏 ⭐留言 &#x1…

Mybatis进阶4-权限管理

权限管理 1.权限 //相当于 职责 2.用户 //相当于 职员&#xff08;职员就职于一个职位&#xff09; 3.角色 //相当于 职位&#xff08;有多个职责&#xff09; 权限管理基础表&#xff1a;权限表&#xff0c;用户表&#xff0c;角色表 问题1&#xff1a;…

Unity 合并子物体获得简化Mesh

合并子物体获得简化Mesh &#x1f959;环境&#x1f96a;Demo &#x1f959;环境 PackageManager安装Editor Coroutines 导入插件&#x1f448; &#x1f96a;Demo 生成参数微调&#xff1a;Assets/EasyColliderEditor/Scripts/VHACDSettings/VHACDSettings.asset

如何做好一个活动策划?

活动策划的关键要素是什么&#xff1f; 首先&#xff0c;要明确一个概念:做活动就是走钢丝&#xff0c;没有保险的高空走钢丝!因为&#xff0c;活动没有“彩排”&#xff0c;只有现场"直播”! 无论什么类型的活动&#xff0c;人数是50人还是2000人&#xff0c;也不论预算…

我的256天之创作纪念日

目录 时光 数据的一些变化 开心的事 憧憬 时光 自上次CSDN的消息推送&#xff0c;又一个128天过去了&#xff0c;整天的工作和生活都在忙忙碌碌中度过&#xff0c;每到能静下来片刻&#xff0c;都倍感珍惜。因为一些原因&#xff0c;能够陪伴家人的时间越来越少&#xff…

[译]Elasticsearch _source Doc_values And Store Performance

原文地址 https://sease.io/2021/02/field-retrieval-performance-in-elasticsearch.html 在这篇博文中&#xff0c;我想从性能的角度探讨 Elasticsearch 为我们存储字段和查询时检索字段提供了哪些可能性。 事实上&#xff0c;Lucene&#xff08;Elasticsearch 和 Solr 构建的…

SOL链DApp智能合约代币质押挖矿分红系统开发

随着区块链技术的不断发展和普及&#xff0c;越来越多的项目开始探索基于区块链的去中心化应用&#xff08;DApp&#xff09;。Solana&#xff08;SOL&#xff09;作为一条高性能、低成本的区块链网络&#xff0c;吸引了众多开发者和项目&#xff0c;其中包括了各种类型的DApp&…