2307|0

6802

帖子

0

TA的资源

五彩晶圆(高级)

楼主
 

蓝牙模块HC-05与51单片机的连接 [复制链接]

本帖最后由 Jacktang 于 2020-11-4 07:41 编辑

实验结果
手机安装HC-PDA-ANDROID.apk软件后,开启系统。手机打开蓝牙可以搜索到蓝牙芯片并可以连接,连接码为1234。系统开启后数码管全0,继电器低电平,LED灯不亮。

手机连接到系统后,扫描二维码,得到一串数字后,输入该串密码,得以解锁继电器。输入密码,支持断点续传密码,如本次输入“123”,再次输入“456”,即可完成输入“123456”。输入错误可以选择按键S5清空输入后重新输入。单片机暂时内部设置密码为“12345678”。输入密码的过程中,流水灯会展示输入密码的最后一个字符的ASCII码,用于指示传输过程中字符是否正确被接收。如果输入了错误的密码,则会返回一串错误提示消息,并将数码管清空。输入错误的字符后,流水灯全亮。

输入正确的密码后,继电器高电平,LED构成回路后点亮,返回给手机端计费信息和成功解锁提示。数码管开始计时,每10毫秒变动一次,8个数码管两个一组,分别显示小时、分钟、秒、十毫秒。当处于计费状态时,手机端发送数据,返回无法发送的字样。按下按键S4后,停止计费,返回到手机端计时时长和本次费用,数码管清零,等待下次解锁。

二.单片机工作原理

51单片机有P0、P1、P2、P3四个端口。本次实验将其中P0用于数码管的显示数字,P1用于流水灯的显示,P2端口的第1位(P2^1)接在继电器的DIO端,在单片机内控制高低电平。P3端口的第0位连接蓝牙的TX端,是单片机串口的接收端,P3端口的第1位连接蓝牙的RX端,是单片机串口的发送端。按键S4在内部连接P3端口的第2位,为外部中断0。按键S5在内部连接P3端口的第3位,为外部中断1。开启定时器0,设置好计数器的初始值;外部中断0,1打开;串口中断打开,设置好波特率9600,0,0与蓝牙模块一致即可。

串口的中断触发后,需要软件清除RI(接收中断)与TI(发送中断)值为0,SBUF存放串口接收数据或发送数据,在C语言代码中赋值相同寄存器,在物理上分为发送和接收,每次1字节缓冲。

定时器0设置工作方式0,13位计数器,计数到8192。晶振为11.0592MHz,每个机器周期需要12个时钟周期,计数5000次,所以每次进入中断的时间为5000*12/11.0592M=0.00543s,所以每次进入中断时间为0.005秒,数码管需要每10ms进入一次,所以每次加到2的时候,数码管变动。计数器初值为(8192-5000),分别存放入TH0与TL0,高低位。

外部中断直接设置触发方式ITx=0/1低电平触发或下降沿触发后,开启外部中断,EXx=1后,编写相应中断函数即可。

蓝牙模块首先按住复位键上电,即进入AT指令模式,对它输入AT指令进行设置名字、串口波特率、主从回环等后,连接到单片机上即可使用。

三.模块工作原理

二维码扫描APP:

通过谷歌开源ZXing库开发了一款安卓APP,并安装到手机,扫描出数据后通过蓝牙模块输入到单片机中。

扫描后为“12345678”。


蓝牙模块HC-05:

TX连接单片机P3.0口,RX连接单片机P3.1口。

在蓝牙模块连接到单片机上前,首先通过USB-TTL转接器,连接到电脑上后,通过串口调试助手调试。首先进入命令调试模式,输入AT指令,设置模块的参数。

设置蓝牙的名称,用指令

AT+name=”LiMou”\r\n          设置蓝牙模块名字为LiMou,方便后续查找。

设置自动连接模式的串口波特率为9600,用指令

AT+uart=9600,0,0\r\n             设置波特率为9600,停止位1位,无校验位。

用于手机与单片机之间的通信者,发送密码到密码上,在接入计费系统时,向手机端发送解锁成功标志,并提供计费标准。解锁失败发送解锁失败指示。

测试问题:

如果串口收不到数据,换一个模块。

如果单片机收到数据错误,调整波特率,通过串口助手调节。

继电器模块:

       继电器电源连接VCC,GND接地,DIO连接单片机P2.1端口,继电器的模块的开闭表示单车的开闭锁。在继电器下接入一个LED的小灯,用它的亮灭来表示继电器是否上电,是否已经开锁。

       LED:

连接通过继电器构成回路,长脚(正极)接电源,短脚(负极)接继电器常闭端,继电器公共触点引到地,构成回路,点亮小灯(在继电器DIO高电平时)。


数码管:

       给P1口送编码即可,0-F的编码分别为0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71。

其中P2^6为段选通,P2^7为位选通。

用于计时模块和指示通过蓝牙传入的密码数据。

按键:

       根据不同按键连接的端口触发不同的中断,编写对应的中断函数。在本系统中,外部中断0通过独立键盘模块按键S4实现,低电平触发,在计费状态下,停止计费,并向手机发送计费结果。

       外部中断1通过独立键盘模块按键S5实现,下降沿触发,当在非计费状态下,清空数码管显示为全0。

      

流水灯:


流水灯连接到单片机的P1端口,直接对P1口赋值即可,高电平为灭,低电平亮。用于指示接收数据的ASCII码和错误输入提示,全亮。


51单片机代码:

#include "reg51.h"
#include <intrins.h>
sbit lock = P2 ^ 1;
sbit dula = P2 ^ 6;
sbit wela = P2 ^ 7;
 
unsigned char digit_led_pointer = 0;
unsigned char count = 0;
unsigned char opened = 0;
 
unsigned char code table[] = { 0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d, 0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71 };    // 数码管0-f表
 
unsigned char led[] = { 0, 1, 2, 3, 4, 0, 0, 0 };    //存放8个数码管显示的数字
 
 
void delay(unsigned int i)        //延迟函数
{
    while (--i);
}
 
void clear_digit(void)            //清空数码管,全变成0
{
    unsigned char i = 0;
    for (; i < 8; ++i)
        led = 0;
    digit_led_pointer = 0;
}
 
 
void send(unsigned char dat)    //通过串口发送到蓝牙模块上,蓝牙模块发送出去一个字节的字符
{
    ES = 0;
    SBUF = dat;
    while (!TI);
    TI = 0;
    ES = 1;
}
 
void send_str(unsigned char * str)    //通过蓝牙模块发送一个字符串
{
    while (*str != '\0')
        send(*str++);
}
 
 
void display(unsigned char dig)        //输入密码阶段的显示,在已经输入的密码后续进行输入
{
    led[digit_led_pointer++] = dig;
    if (digit_led_pointer == 8)
    {
        //判断如果输入密码为12345678的话,返回解锁成功
        if (led[7] == 8 && led[6] == 7 && led[5] == 6 && led[4] == 5 && led[3] == 4 && led[2] == 3 && led[1] == 2 && led[0] == 1 && (opened == 0))
        {
            //开锁状态置位
            opened = 1;
            lock = 1;    //开锁继电器,高电平,点亮LED
            clear_digit();
            send_str("price: 1Yuan/min\n");
            send_str("lock successfully!\n");
        }
        else
        {
            delay(10000000);    //输入8位密码后,延迟显示错误密码
            clear_digit();
            //解锁失败
            send_str("lock fail!\n");    //向手机发送提示信息
        }
        digit_led_pointer = 0;
    }
}
 
 
//计时模块,每次进入该函数,为10毫秒
void time_add_10ms()
{
    led[7]++;
    //100进位,每次10毫秒
    if (led[7] == 10)
    {
        led[7] = 0; led[6]++;
    }
 
    if (led[6] == 10)
    {
        led[6] = 0; led[5]++;
    }
 
    //60进位,为秒数
    if (led[5] == 10)
    {
        led[5] = 0; led[4]++;
    }
 
    if (led[4] == 6)
    {
        led[4] = 0; led[3]++;
    }
    //分钟数
    if (led[3] == 10)
    {
        led[3] = 0; led[2]++;
    }
 
    if (led[2] == 10)
    {
        led[2] = 0; led[1]++;
    }
    //小时数,如果100个小时清零
    if (led[1] == 10)
    {
        led[1] = 0; led[0]++;
    }
 
    if (led[0] == 10)
        clear_digit();
}
 
//定时器0,为1号中断
void timer0_ISR(void) interrupt 1
{
    if (opened == 1)
    {
        //晶振为11.0592MHz,每个机器周期需要12个时钟周期,计数5000次,所以每次进入中断的时间为
        //    5000*12/11.0592M=0.00543s,每次进入add_10ms函数需要计数2次,所以每次count==2的时候,数码管显示+1
        TL0 = (8192 - 5000) % 32;
        TH0 = (8192 - 5000) / 32;
        count++;
        if (count == 2)
        {
            //达到10ms了
            time_add_10ms();
            count = 0;
        }
    }
}
 
 
 
void uart() interrupt 4
{
    unsigned char ky;
 
    if (RI)
    {
        RI = 0;        //需要软件清除标志位
        ky = SBUF;    //接收到的数据保存到
        if (opened == 1)
        {
            send_str("can't stop, please press the button S4!\n");    //计费状态不接受输入,并返回提示信息。
            return;
        }
        ES = 0;
        P1 = ky;    //data send to Port 1,通过流水灯显示接收到的数据的ASCII码
 
        //根据收到的数据进行异常处理或数码管输入
        if (ky >= '0'&&ky <= '9')
        {
            display(ky - '0');
        }
        else if (ky >= 'a' && ky < 'f')
        {
            display((unsigned char)((ky - 'a') + 10));
        }
        else
        {
            //错误数据,流水灯全亮。
            P1 = 0x00;
        }
        ES = 1;
    }
}
 
void init_uart()
{
    TMOD = 0x25;            //定时器1工作方式2,计数器0工作方式1
    SCON = 0x50;            //串口工作方式1
    EA = 1;             //开总中断
    ES = 1;             //开串口中断
    TH1 = 0xfd;            //串口波特率9600
    TL1 = 0xfd;
    TR1 = 1;             //定时器1工作    
}
 
void timer0_init(void)
{
    TMOD &= 0xF0;    //计数器0方式0
    TL0 = (8192 - 5000) % 32;    //计数5000次
    TH0 = (8192 - 5000) / 32;
    count = 0;
    ET0 = 1;        //开启计时器和中断
    TR0 = 1;        
}

 
void init_EX(void)
{
    EX0 = 1;
    IT0 = 0;
    EX1 = 1;        //开启外部中断 0
    IT1 = 1;          //设置成低电平触发,1为下降沿触发
}
 
//把计时费用转换成字符串发送给手机端
void num2str(int cost)
{
    while (1)
    {
        if (cost / 10 != 0)
            send(cost / 10 + '0');
        else
        {
            send(cost + '0');
            break;
        }
        cost %= 10;
    }
}
 
//外部中断0,S4按下停止计费
void Ex0_IRQ(void) interrupt 0
{
    //不在计费状态下没反应
    if (opened == 1)
    {
 
        int time;
        opened = 0;
        send_str("duration:");
        send(led[0] + '0'); send(led[1] + '0'); send(':');
        send(led[2] + '0'); send(led[3] + '0'); send(':');
        send(led[4] + '0'); send(led[5] + '0');
        send('\n');
        time = ((led[0] + led[1]) * 60 + (led[2] * 10 + led[3]));
        send_str("cost: RMB "); num2str(time + 1); send('\n');
        lock = 0;
        clear_digit();
    }
}
 
//外部中断1,2号中断,S5按下后清空数码管
void Ex1_IRQ(void) interrupt 2
{
    //不在计费状态下才有反应
    if (opened == 0)
    {
        digit_led_pointer = 0;
        clear_digit();
    }
}
 
void main()
{
    unsigned char num = 0;
    //初始化
    init_uart();
    timer0_init();
    clear_digit();
    init_EX();
    lock = 0;
    while (1)
    {
        //显示数码管
        for (num = 0; num < 8; ++num)
        {
            wela = 1;
            P0 = _crol_(0xfe, num);    //选中数码管,循环左移
            wela = 0;
            dula = 1;
            P0 = table[led[num]];
            delay(100);
            dula = 0;
        }
    }
}

此帖出自无线连接论坛
点赞 关注
 

回复
举报
您需要登录后才可以回帖 登录 | 注册

随便看看
查找数据手册?

EEWorld Datasheet 技术支持

相关文章 更多>>
关闭
站长推荐上一条 1/9 下一条

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

About Us 关于我们 客户服务 联系方式 器件索引 网站地图 最新更新 手机版

站点相关: 国产芯 安防电子 汽车电子 手机便携 工业控制 家用电子 医疗电子 测试测量 网络通信 物联网

北京市海淀区中关村大街18号B座15层1530室 电话:(010)82350740 邮编:100190

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2025 EEWORLD.com.cn, Inc. All rights reserved
快速回复 返回顶部 返回列表