2021-04-25

                                                                                                     基于51单片机的蓝牙智能台灯

 

目录

                                                                                                     基于51单片机的蓝牙智能台灯

 

一、功能:

二、系统组成:

三、设计思路:

四、硬件:

      1. 51最小系统板(学过51的都基本了解,这里不做赘述)

      2. 光敏电阻

     3. ADC0832

4. HC-05蓝牙模块

5.超声波测距模块HC-SR04

6.双向可控硅

五、软件

1.电路原理图

2.源代码

六、遇到的问题

七、实物图


一、功能:

      1.当人靠近的时候,灯会自己亮,当人离开的时候,灯会自己灭

       2.靠的太近的时候,蜂鸣器会发出报警

      3.可以连接手机蓝牙,进行亮度调光,发送1,最亮,;发送2,较亮;发送3,最暗;发送4,灭

      4.切换模式,可以随着环境亮度自动调光

二、系统组成:

      51最小系统板+光敏电阻及AD模块+蓝牙模块+超声波测距模块+继电器+可控硅+电源

三、设计思路:

      利用超声波测距,设定不同的距离范围,执行不同的操作,人坐在台灯前,靠的太近,离开,分别对应不同的范围,实现灯的亮灭和报警。可控硅可以实现用小功率控制大功率的效果,在可控硅的的触发脚并联不同阻值的电阻,由四路继电器连接,蓝牙给51单片机发送数字,控制继电器的通断,实现不同亮度的调节。对于自动调光,可采用光敏电阻和ADC0832实现,不同的光照,光敏电阻的阻值不同,利用ADC0832将模拟量转化成数字量,设定不同的范围,执行不同的操作。

四、硬件:

      1. 51最小系统板(学过51的都基本了解,这里不做赘述)

      2. 光敏电阻

                光敏电阻的工作原理是基于内光电效应。当光敏电阻受到一定波长范围的光照时, 它的阻值(亮电阻)急剧减少, 电路中电流迅速增大。 一般希望暗电阻越大越好, 亮电阻越小越好,此时光敏电阻的灵敏度高。 实际光敏电阻的暗电阻值一般在兆欧级, 亮电阻在几千欧以下

     3. ADC0832

2021-04-25_第1张图片2021-04-25_第2张图片

(1)引脚功能

CS:片选使能,低电平使能

CH0:模拟输入通道0

CH1:模拟输入通道1

GND:芯片参考0电位

VCC:电源输入

CLK:芯片时钟输入

DI:信号数据输入,选择通道控制

DO:信号数据输出,转换数据输出

(2)使用方法

         ADC0832芯片与单片机在正常情况下应有4条数据线相连,分别是CLK和CS、D1和D0。但在实际通信时D1端与DO端未同时有效,并且双向连接到单片机,所以在设计电路时可将D1和DO接在一个单片机管脚上进行使用。本设计中ADC0832芯片的D1端和DO端均接到了单片机的P0^0口,选择数据信号从D1端进行输入,数据由D0端进行输出;CLK端接到了单片机的P0^1口为AD芯片提供时钟输入;使能端CS接到了单片机的P0^2口;传感器模拟电压输出到通道CH0端

4. HC-05蓝牙模块

 

2021-04-25_第3张图片2021-04-25_第4张图片

 

(1)VCC:当然这个引脚是接电源的正极,电压的范围为3.3v到5.0v

 

(2)GND:直接接地

(3)TXD:模块串口发送引脚(TTL电平,不能直接接RS232电平),可直接接单片机的RXD引脚

(4)RXD:模块串口接收引脚(TTL电平,不能直接接RS232电平),可直接接单片机的TXD引脚

(5)KEY:用于进入AT状态:高电平有效(悬空默认为低电平)

(6)蓝牙AT指令

 

a.  AT+RESET:该指令顾名思义,是复位HC05蓝牙设备的,蓝牙模块复位后,当然原来连接也就断开了,模块处于INITIALIZED状态,还需要注意的是,如果在用MCU的串口给蓝牙模块发送复位命令,一定要注意发送复位命令1s后才能继续发送其他命令,因为如果你1s内发送其他命令,此时蓝牙模块可能还在复位中,没法响应命令。当然,如果你是用串口工具发送,一般发送两个命令的间隔不会太短。

>AT+RESET\r\n

OK

b.  AT+INIT:该指令初始化SPP规范库,所谓的SPP就是蓝牙串口端口协议,总之,没有初始化SPP库就没发扫描周围的蓝牙设备,没有与摸个蓝牙设备连接。,所以,在应用程序中,一定要初始化SPP库,没有初始化SPP库,就发送扫描指令或连接指令,蓝牙模块会返回ERROR:(16)错误,表示没有初始化SPP库。还有个要注意的是,如果发送多次该指令,蓝牙模块会放回ERROR:(17)重复初始化错误。还有,每次断电后再上电或蓝牙模块复位后,都要重新初始化SPP库。

>AT+INIT\r\n

OK          //正确

FALL  //失败

ERROR:(17)  //重复初始化

 

c.  AT+ROLE:该指令用于选择HC05蓝牙模块的角色,总共有三种角色:master,slave,loop-slave.

AT+ROLE=0\r\n  将蓝牙模块设置成从角色,只能被动连接

AT+ROLE=1\r\n  将蓝牙模块设置成主角色,可以查询周围SPP蓝牙从设备,并发送连接

AT+ROLE=2\r\n  将蓝牙模块设置成回环角色,被动连接,接收远程蓝牙模块主设备数据并将数据原样返回给远程蓝牙设备

   这几个指令用在不同的场合,当设置蓝牙模块为从设备的时候,可以用手机的相关蓝牙装串口软件连接该设备,进行通讯;当设置成主角色的时候,可以搜索周遭的蓝牙从设备,并连接,这种模式在应用中很常用;回环角色很多时候都是用来做测试用的。

>AT+ROLE=1\r\n

OK

 

d. AT+PSWD:这个指令时设置蓝牙模块的配对密码,蓝牙模块在做从模块的时候,如果用手机要连接该蓝牙设备,就要键入蓝牙的配对密码才能连接,比如说HC05默认的配对密码是1234,那么就要在手机的输入该配对密码1234才能连接我们的HC05

>AT+PSWD=1234\r\n

OK

 

e.  AT_UART:这个指令是设置串口的参数,指令的格式如下:

AT+UART=,,\r\n

Param:波特率,一些常用的波特率都可以设置

Param1:停止位,一般设置成0,表示为1个停止位

Param2:校验位,一帮设置为0,表示不用校验

该指令是设置蓝牙模块与蓝牙模块之间通讯时,蓝牙模块的串口参数。HC05模块在默认的配置下是设置成9600,也就是说,在AT模式下,我们用38400与HC05通讯,而在HC05与某蓝牙模块通讯时,则我们用9600的波特率接收HC05从蓝牙模块的接收到的数据。这里建议将波特率改为38400,这是因为,当用串口调试工具连接HC05时,AT模式是用38400波特率,而连接后默认是9600,这样的不断的切换串口调试工具的波特率,会很麻烦,所以设置成38400后,就方便调试了。

>AT+UART=38400,0,0\r\n

OK

 

f.  AT+INQM:设置或查询访问模式,格式如此:

AT+INQM=,,\r]n

Param:0——inquiry_mode_standard,1——inquiry_mode_rssi,表示标准查询还是带信号强度的查询。

Param1:最多蓝牙设备响应的数量

Param2:最大查询时间(1~48,折合成时间,1,28s~61.44s)

将这个指令是为扫描指令做铺垫,根据自己的实际情况调整。

>AT+INQM=1,1,15

OK

g.  AT+INQ:查询蓝牙设备,返回的格式如下:

+INQ:,,....

Param:蓝牙地址

Param1:设备类

Param2:RSSI信号强度

举个例子:+INQ:98D3:31:500DF8,1F00,7FFF,

98D3:31:500DF8表示蓝牙的地址,这里有需要补充下蓝牙的相关知识,蓝牙地址的由NAP(24位地址低端部分):UAP(8为地址高端部分):LAP(16为无意义地址部分),所以该地址:98D3表示LAP,31表示UAP,500DF8表示LAP

>AT+INQ 

+INQ:98D3:31:500DF8,1F00,7FFF //有的话列出 

OK

 

h.  AT+RNAME:这个指令获得远端蓝牙设备的名字,我们手机上看到的就是这个名字,而不会直接给出蓝牙设备地址

>AT+RNAME? 98D3,31,500DF8   //主要这里是逗号,而不是冒号

+RNAME:EST527

OK

i.  AT+LINK:这个命令连接远程设备蓝牙,其实没有什么好说的,连接上后,LED引脚输出高电平,如果该引脚有接上拉LED灯,则会发现LED灭了。

>AT+LINK=98D3,31,500DF8

OK

j.  AT+STATE:这个指令用来查询蓝牙模块当前的状态,状态种类如下:

INITIALIZED——初始化状态

READY——准备状态

PAIRABLE——可配对状态

PAIRED——配对状态

INQUIRING——查询状态

CONNECTING——正在连接状态

CONNECTED——连接状态

DISCONNECTED——断开连接

UNKNOW————位置状态

 

>AT+STATE

+STATE:INITIALIZED

OK

 

+STATE:PAIRED

OK

k.  AT+CMODE:这个指令其实很重要。

AT+CMODE=0\r\n  指定蓝牙地址连接模式(指定蓝牙地址呦绑定指令设置)

AT+CMODE=1\r\n  任意蓝牙地址连接模式(不受绑定指令的设置地址约束)

AT+CMODE=2\r\n  回环角色

为0时,该指令设置模块为指定地址配对,如果先设置模块为任意地址,然后配对,接下去使用该指令,则模块会记忆最后一次配对的地址,下次上电会一直搜索该地址的模块,直到搜索到为止。

 为1时,该指令设置模块可以对任意地址的蓝牙模块进行配对,只要有模块的配对秘钥跟自己的一样的就能自动连上。

AT+CMODE=1\r\n

OK

(7)测试代码

 #include 
#include
#include 
#define uchar unsigned char 
#define uint unsigned int
uchar aa,bb; 
uchar Buffer[4] = {0};
uint  xms,i,j;
sbit led1=P1^0;
sbit led2=P1^1;
sbit led3=P1^2;
void delayms(uint xms)
 {
 uint i,j;
 for(i=xms;i>0;i--)
 for(j=110;j>0;j--);
 }
void Com_Int(void) interrupt 4
{

EA = 0;
if(RI == 1)   //当硬件接受到一个数据时,RI会置位
{
Buffer[0] = SBUF - 48; //这里减去48是因为从电脑发送过来的是ASCII码
RI = 0; 

}
EA = 1;

/********************************************************************
串口初始化
***********************************************************************/
void Com_Init(void)
{
TMOD = 0x20;
PCON = 0x00;
SCON = 0x50; 
TH1 = 0xFd; //波特率9600
TL1 = 0xFd;
TR1 = 1; //启动定时器1
ES = 1; //开启串口中断
EA = 1; //开启总中断
}
/**************************************************************************
主函数
**************************************************************************/

void main()
{
 delayms(100);

 Com_Init();

while(1)
{
  

switch(Buffer[0])
{

case 1: led1=1; break;
case 2: led1=0 ; break;
default:break; 


}  
 

5.超声波测距模块HC-SR04

2021-04-25_第5张图片

(1)引脚:VCC、Trig、Echo、GND

        VCC接电源正极

    GND接电源负极

    Trig是传感器触发信号输入引脚---P3^2

     Echo是传感器测距数据输出引脚--P3^3  

(2)工作原理

        单片机向模块的Trig引脚发出至少10μs的高电平信号以触发传感器测距。模块自动发送8个40kHz的方波,模块自动检测是否有信号返回。

当有信号返回时,通过IO输出一高电平,该高电平持续的时间就是超声波从发射到返回的时间。时间单位 一般为微秒(μs),声速约为344m/s,

则可推导:

  测试距离(m)= [高电平时间(μs)×10^(-6)×声速(344m/s)]/2

           = 高电平时间(μs)×172/10^6

  此时的测试距离的单位为米,转换为厘米则

  测试距离(cm)= 高电平时间(μs)×172/10^6×100

             = 高电平时间(μs)/58

6.双向可控硅

2021-04-25_第6张图片

(1)引脚

1------T1主电极       

2------T2主电极

3------G门极

(2)工作原理

    双向可控硅是两个方向都可以导通的,所以就不区分阳极和阴极。只要在G极注入正向电压就可以让T1和T2极导通了,非常适用于交流电工作的器件的控制。可控硅一旦导通,不管G脚的电压如何变,所控制的器件状态保持不变,要想改变状态,只有切断总电源

五、软件

1.电路原理图

2021-04-25_第7张图片

说明:继电器的com端和火线相连,常开端和电阻相连,IN脚连接单片机的IO口,当给其一个低电平,继电器导通,NO端和com端的电平将相同,及相当于NO端也连接到火线。可控硅导通,灯泡点亮,因为电源是交流电,当到过零电压,灯泡熄灭。可控硅G脚高电压保持的时间跟相连的电阻有关,这样可以通过单片机控制不同的继电器,从而实现调光。

2.源代码

#include "1602.h"

#define uchar unsigned char 
#define uint unsigned int

#define VELOCITY_30C    3495     
#define VELOCITY_23C    3453    
uchar aa,bb; 
uchar Buffer[4] = {0};
uint  xms,i,j;


void delayms(uint xms)
 {
 uint i,j;
 for(i=xms;i>0;i--)
 for(j=110;j>0;j--);
 }

sbit INPUT= P3^3;             
sbit OUTPUT= P3^2;           

sbit DW1=P1^4;          
sbit DW2=P1^5;          
sbit DW3=P1^6;          
sbit Beep=P1^3;          
sbit rw    = P1^1;

long int distance=0;           
uchar table[]="    Welcome to   ";
uchar table0[]="      yfmcu      ";
uchar table1[]="There's no echo.";
uchar table2[]="      yfmcu      ";
uchar table3[]="Distance:";
uchar count;

extern void write_date(uchar date);
extern void write_com(uchar com);
extern void delay(uint x);
/******************************************************************************/

/******************************************************************************/
void Delay_xMs(unsigned int x)
{
    unsigned int i,j;
    for(i = 0;i < x;i++ )
    {
        for(j = 0;j < 3;j++ )
        {
            ;
        }
    }
}

/******************************************************************************/

/******************************************************************************/    
void delayt(uint x)
{
    uchar j;
    while(x-- > 0)
    {
          for(j = 0;j < 125;j++)
        {
            ;
        }
    }
}
/******************************************************************************/

/******************************************************************************/
void Init_MCU(void)
{
    TMOD = 0x01;     
     TL0 = 0x66;
    TH0 = 0xfc;        
    ET0 = 1;        
    EA = 1;         
}
/******************************************************************************/

/******************************************************************************/
void Init_Parameter(void)
{
     OUTPUT =1;
     INPUT = 1;
     count = 0;
     distance = 0;
}

/******************************************************************************/

/******************************************************************************/
void Trig_SuperSonic(void)
{
     OUTPUT = 1;
     delayt(1);
     OUTPUT = 0;
}
/******************************************************************************/

/******************************************************************************/
void Measure_Distance(void)
{
    uchar l;
    uint h,y;
    TR0 = 1;
    while(INPUT)
    {
        ;
    }    
    TR0= 0;
    l = TL0;
    h = TH0;
    y = (h << 8) + l;
    y = y - 0xfc66;
    distance = y + 1000 * count;
    TL0 = 0x66;
    TH0= 0xfc;
    delayt(30);
    distance = VELOCITY_30C * distance / 90000;
    
            
}
void Com_Int(void) interrupt 4
{

EA = 0;
if(RI == 1) 
{
Buffer[0] = SBUF - 48; 
RI = 0; 

}
EA = 1;

/********************************************************************
´®¿Ú³õʼ»¯
***********************************************************************/
void Com_Init(void)
{
TMOD|= 0x20;
PCON = 0x00;
SCON = 0x50; 
TH1 = 0xFd; 
TR1 = 1; 
ES = 1; 
EA = 1;
}
void light_l()
{
    
    
     /*if(Buffer[0]==1||distance>200&&distance<700)
        
    {
    
        DW4=0;
        DW1=0;
        DW2=1;
        DW3=1;
    
    
        
        
    }
    
    
    else if(Buffer[0]==2)
    {
      DW4=0;
        DW1=1;
        DW2=0;
        DW3=1;
        
        
    
        
    }
     else if(Buffer[0]==3)
    {
    
        DW4=0;
        DW1=1;
        DW2=1;
        DW3=0;
        
        
        
        
    }
    else if(Buffer[0]==4||distance>700)
    {
        DW4=1;
        DW1=1;
        DW2=1;
        DW3=1;
    }*/

        if(distance>700)
        
    {
    
      Beep=1;
        DW1=1;
        DW2=1;
        DW3=1;
        if(Buffer[0]==1||distance>200&&distance<700)
        {
        
        DW1=0;
        DW2=1;
        DW3=1;
    
        }
        
            
    
    
        
        
    }
    
    
    else if(Buffer[0]==2)
    {
    
        DW1=1;
        DW2=0;
        DW3=1;
        
        
    
        
    }
     else if(Buffer[0]==3)
    {
    
        
        DW1=1;
        DW2=1;
        DW3=0;
        
        
        
        
    }
    

}
/*void jiance()
{
    int flag=0;
    if(distance>200&&distance<700)
        flag=0;
    else 
        flag=1;
        
}
void light_l()
{
    switch(Buffer[0]-flag)
    {
        case 1:
            
        DW4=0;
        DW1=0;
        DW2=1;
        DW3=1;
        break;
        case 2:
            
        DW4=0;
        DW1=1;
        DW2=0;
        DW3=1;
        break;
        case 3:
        DW4=0;
        DW1=1;
        DW2=1;
        DW3=0;
        break;
    }
        
        
            
            
            
}*/
    

    
void main()
{
     
    delayms(100);

  Com_Init();
     rw = 0;
    
    //Init_MCU();
  Init_Parameter();

    Delay_xMs(30000);
    

while(1)
{
  
       Trig_SuperSonic();        
         while(INPUT == 0)         
         {
             ;
         }
         Measure_Distance();        
            
     light_l();
          if(distance>0&&distance<100)
             Beep=0;
             else Beep=1;
         Init_Parameter();      
         delayt(100);        
}  


/******************************************************************************/
/******************************************************************************/

void timer0 (void) interrupt 1
{
    TF0 = 0;
    TL0 = 0x66;
    TH0 = 0xfc;
    count++;
    if(count == 18)
    {
        TR0 =0;
        TL0 = 0x66;
        TH0 = 0xfc;
        count = 0;
    }
}
/******************************************************************************/

六、遇到的问题

1.当超声波避障和ADC0832,蓝牙串口同时工作,会导致定时器紊乱

解决方法:用两个单片机,将ADC0832单独放在一个单片机上控制,把一个设置为主机,一个设置为从机,实现两个单片机之间的通信

七、实物图

 

 

 

         

 

 

 

 

你可能感兴趣的:(单片机,传感器,嵌入式)