基于51单片机超声波的停车场车位管理系统proteus仿真原理图PCB

功能介绍:
0.本系统采用STC89C52作为单片机
1.LCD1602液晶实时显示当前车位状态
2.当超声波探头检测到距离小于500mm时,判断为当前车位有车,否则无车
3.超声波传感器型号采用HC-SR04
4.预留4个功能按键和蓝牙通信模块可自行开发新的功能
5.电源接口采用DC002直接输入5V供电

原理图:
在这里插入图片描述

PCB:
在这里插入图片描述

主程序:

#include <reg52.h>
#include <intrins.h>
#include <stdio.h>
#include "delay.h"
#include "lcd1602.h"

#define OFF 1
#define ON 0

sbit TRIG_1 = P1^0; //接口定义
sbit ECHO_1 = P1^1; //接口定义
sbit LED_VOID_1 = P1^2;
sbit LED_PARKING_1 = P1^3;

sbit TRIG_2 = P2^0; //接口定义
sbit ECHO_2 = P2^1; //接口定义
sbit LED_VOID_2 = P1^4;
sbit LED_PARKING_2 = P1^5;

sbit TRIG_3 = P2^4; //接口定义
sbit ECHO_3 = P2^5; //接口定义
sbit LED_VOID_3 = P2^3;
sbit LED_PARKING_3 = P2^2;

xdata unsigned char dis0[16]; //定义显示区域临时存储数组
xdata unsigned char dis1[16];

bit disFlag = 0; //显示标志
bit refreshFlag = 0; //显示标志
unsigned char i;

unsigned char occupiedFlag1 = 0; //位置标志
unsigned char occupiedFlag2 = 0; //位置标志
unsigned char occupiedFlag3 = 0; //位置标志
unsigned char occupiedNum = 0;     //占用总数

float f_distance1 = 0;		 //距离1
float f_distance2 = 0;		 //距离2
float f_distance3 = 0;		 //距离3

void Timer1_Init(void); //函数声明
void Measuring(void);
//void UART_SendStr(unsigned char *s, unsigned char length);
//void UART_Init(void);
//void UART_SendByte(unsigned char dat);


void main(void)
{
    
    
    TRIG_1 = 0;
    TRIG_2 = 0;
    TRIG_3 = 0;
    
    TMOD &= 0xF0;                //使用模式1,16位定时器,使用"|"符号可以在使用多个定时器时不受影响
    TMOD |= 0x01;                //使用模式1,16位定时器,使用"|"符号可以在使用多个定时器时不受影响
    
    TH0 = 0; //重新赋值 20ms
    TL0 = 0;
    EA = 1;  //总中断打开
    Timer1_Init(); //定时器0初始化

    LCD_Init();  //初始化液晶
    DelayMs(20); //延时有助于稳定
    LCD_Clear(); //清屏

    while (1)
    {
        if (refreshFlag == 1)
		{
			refreshFlag = 0;
            Measuring();
        }
        if (disFlag == 1) //定时显示
        {
            disFlag = 0; //标志位清零

            if (f_distance1 <= 500) //检测到车距0.5m
            {
                occupiedFlag1 = 1;
            } //标志
            else
            {
                occupiedFlag1 = 0;
            }

            if (f_distance2 <= 500) //检测到车距0.5m
            {
                occupiedFlag2 = 1;
            } //标志
            else
            {
                occupiedFlag2 = 0;
            }

            if (f_distance3 <= 500) //检测到车距0.5m
            {
                occupiedFlag3 = 1;
            } //标志
            else
            {
                occupiedFlag3 = 0;
            }

            if (occupiedFlag1 == 1)
            {
                LCD_DispStr(0, 1, "P ");
                LED_VOID_1 = OFF;
                LED_PARKING_1 = ON;
            } //显示占用
            else
            {
                LCD_DispStr(0, 1, "N ");
                LED_VOID_1 = ON;
                LED_PARKING_1 = OFF;
            }

            if (occupiedFlag2 == 1)
            {
                LCD_DispStr(2, 1, "P ");
                LED_VOID_2 = OFF;
                LED_PARKING_2 = ON;
            } //显示占用
            else
            {
                LCD_DispStr(2, 1, "N ");
                LED_VOID_2 = ON;
                LED_PARKING_2 = OFF;
            }

            if (occupiedFlag3 == 1)
            {
                LCD_DispStr(4, 1, "P ");
                LED_VOID_3 = OFF;
                LED_PARKING_3 = ON;
            } //显示占用
            else
            {
                LCD_DispStr(4, 1, "N ");
                LED_VOID_3 = ON;
                LED_PARKING_3 = OFF;
            }
            occupiedNum = occupiedFlag1 + occupiedFlag2 + occupiedFlag3;                      //占用总数
            sprintf(dis0, "1 2 3  P:%d  N:%d ", (int)occupiedNum, (int)(3 - occupiedNum)); //打印
            LCD_DispStr(0, 0, dis0);                                           //显示                                          //显示
        }

    }
}

void Measuring(void)
{
    static long cnt = 0; //定时器计数
    
    TR1 = 0;
    
    TRIG_1 = 1; //启动一次模块		//不可以使用其他终端 容易造成死循环
    DelayUs10x(1);
    TRIG_1 = 0;
    while (!ECHO_1)
        ;	 //当RX为零时等待
    TR0 = 1; //开启计数
    while (ECHO_1)
        ; //当RX为1计数并等待
    TR0 = 0;
    cnt = (long)(TH0 * 256 + TL0);
    TH0 = 0;
    TL0 = 0;
    f_distance1 = (float)cnt * 17 / 100.0 * 1.102; //算出来是mm (g_cnt * 340 / 2) / 1000.0 * 1.102;系数
    cnt = 0;

    TRIG_2 = 1; //启动一次模块		//不可以使用其他终端 容易造成死循环
    DelayUs10x(1);
    TRIG_2 = 0;
    while (!ECHO_2)
        ;	 //当RX为零时等待
    TR0 = 1; //开启计数
    while (ECHO_2)
        ; //当RX为1计数并等待
    TR0 = 0;
    cnt = (long)(TH0 * 256 + TL0);
    TH0 = 0;
    TL0 = 0;
    f_distance2 = (float)cnt * 17 / 100.0 * 1.102; //算出来是mm (g_cnt * 340 / 2) / 1000.0 * 1.102;系数
    cnt = 0;

    TRIG_3 = 1; //启动一次模块		//不可以使用其他终端 容易造成死循环
    DelayUs10x(1);
    TRIG_3 = 0;
    while (!ECHO_3)
        ;	 //当RX为零时等待
    TR0 = 1; //开启计数
    while (ECHO_3)
        ; //当RX为1计数并等待
    TR0 = 0;
    cnt = (long)(TH0 * 256 + TL0);
    TH0 = 0;
    TL0 = 0;
    f_distance3 = (float)cnt * 17 / 100.0 * 1.102; //算出来是mm (g_cnt * 340 / 2) / 1000.0 * 1.102;系数
    cnt = 0;
    
    TR1 = 1;
}

void Timer1_Init(void)
{
    TMOD |= 0x10;                //使用模式1,16位定时器,使用"|"符号可以在使用多个定时器时不受影响
    TMOD &= 0x0F;                //使用模式1,16位定时器,使用"|"符号可以在使用多个定时器时不受影响
    TH1 = (65536 - 18432) / 256; //重新赋值 20ms
    TL1 = (65536 - 18432) % 256;
    EA = 1;  //总中断打开
    ET1 = 1; //定时器中断打开
    TR1 = 1; //定时器开关打开
}

void Timer1_Interrupt(void) interrupt 3
{
    static unsigned int time_20ms = 0;

    TH1 = (65536 - 18432) / 256; //重新赋值 20ms
    TL1 = (65536 - 18432) % 256;

    time_20ms++;

    if (time_20ms >= 25)
	{
		disFlag = 1;
		time_20ms = 0;
	}
	else
	{
		time_20ms++;
	}

	if (time_20ms % 5 == 0)
	{
		refreshFlag = 1;
	}
}


仿真演示视频:
https://www.bilibili.com/video/BV15B4y197mW/

实物演示视频:
https://www.bilibili.com/video/BV1vt4y1W7cM/

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值