仿真图:
芯片/模块的特点:
AT89C52简介:
AT89C52是一款经典的8位单片机,是意法半导体(STMicroelectronics)公司生产的一系列单片机之一。它基于8051内核,并具有许多与其兼容的特性。
AT89C52的主要特点如下:
内部存储器:AT89C52具有8KB的闪存(Flash)存储器,可用于存储用户程序和数据。这些存储器的内容可以通过编程器进行编程和擦除。
RAM存储器:AT89C52配备了256字节的随机存取存储器(RAM),用于暂存数据和程序的变量。
外部扩展性:AT89C52支持多种外部扩展设备的连接,包括外部存储器(如RAM、EEPROM)和外设(如ADC、LCD、UART等),通过外部硬件连接,可以扩展单片机的功能和应用。
通用I/O引脚:AT89C52拥有32个可编程的通用输入/输出引脚,可用于连接外部设备和与其他芯片进行通信。
定时器/计数器:AT89C52内置了3个16位定时器/计数器和一个可编程的串行定时器/计数器。这些计时器/计数器可用于实现定时功能、生成脉冲信号、测量时间间隔等。0
串行通信:AT89C52支持串行通信接口,包括UART(串行异步通信)和SPI(串行外设接口),便于与其他设备进行数据通信和交互。
低功耗模式:AT89C52具有多种低功耗模式,如空闲模式和电源下模式,在不需要执行任务的时候可以将CPU进入低功耗状态以节省能量。
宽电源电压范围:AT89C52的工作电压范围通常为4.0V至5.5V,可以满足大多数应用需求。
ADC0832特点:
8位分辨率:ADC0832可以将模拟输入信号转换为8位数字输出。这意味着它可以将模拟信号划分为256个不同的离散电平,提供相对较低的分辨率。
双通道输入:ADC0832具有两个模拟输入通道,使其能够同时转换两个模拟信号。这对于需要同时测量多个信号的应用非常有用。
内部参考电压源:ADC0832提供了一个内部的参考电压源,它可以用作模拟输入信号的参考电压。这样可以简化外部电路设计,并提供更稳定和准确的参考电压。
串行输出:ADC0832通过串行接口(SPI或I2C)输出转换结果。这种串行输出形式使其与微控制器或其他数字设备的通信变得更加简单和方便。
低功耗:ADC0832具有较低的功耗特性,适合在低功耗应用中使用。
可编程时钟频率:ADC0832的转换速度可以通过控制输入时钟频率进行编程。这使得可以根据应用的需求调整转换速度,并平衡转换精度和速度。
内部自校准:ADC0832具有内部自校准电路,可以降低转换误差,并提供更准确的转换结果。
主程序:
#include <reg52.h>
#include <intrins.h>
#include <stdio.h>
#include "lcd1602.h"
#include "delay.h"
#include "tlc0832.h"
sbit FAN = P2^0; //接口定义
sbit BUZZER = P2^2;
sbit RELAY = P2^1;
sbit LED_YELLOW = P1^6;
sbit LED_GREEN = P1^5;
sbit LED_RED = P1^7;
unsigned char dis0[16]; //数组暂存
unsigned char dis1[16]; //数组暂存
unsigned char midVolt; //ad采集电压
unsigned char i;
bit dispFlag = 1;
void Timer0_Init(void); //函数声明
void UART_Init(void);
void UART_SendByte(unsigned char dat);
void UART_SendStr(unsigned char *s, unsigned char length);
void main(void)
{
Timer0_Init(); //定时器0初始化
LCD_Init();
DelayMs(100);
LCD_Clear();
UART_Init();
DelayMs(120);
UART_SendStr("ready ok!\r\n", 10);
LCD_DispStr(0, 0, " Detector ");
while (1) //主循环
{
if (dispFlag == 1)
{
dispFlag = 0;
midVolt = ReadADC(AIN0_GND);
if (midVolt < 66)
{
midVolt = 66;
}
midVolt = 100 * (midVolt - 66) / (255 - 66); //读取AD检测到的值转化为百分比,减去初始值
sprintf(dis0, "CONC. :%2d%%\r\n", (unsigned int)midVolt); //串口发送
UART_SendStr(dis0, 12);
sprintf(dis1, " Percent :%02d%% ", (unsigned int)midVolt); //显示当前检测值
LCD_DispStr(0, 1, dis1);
if (midVolt < 30)
{
FAN = 1; //关 风扇
BUZZER = 1; //关 蜂鸣器
RELAY = 1; //关闭继电器打开气阀
LED_GREEN = 0; //开 绿灯
LED_YELLOW = 1; //关 黄灯
LED_RED = 1; //关 红灯
}
else if (midVolt < 50)
{
FAN = 1; //关 风扇
BUZZER = 1; //关 蜂鸣器
RELAY = 0; //打开继电器关闭气阀
LED_GREEN = 1; //关 绿灯
LED_YELLOW = 0; //开 黄灯
LED_RED = 1; //关 红灯
}
else
{
FAN = 0; //开 风扇
BUZZER = 0; //开 蜂鸣器
RELAY = 0; //打开继电器关闭气阀
LED_GREEN = 1; //关 绿灯
LED_YELLOW = 1; //关 黄灯
LED_RED = 0; //开 红灯
}
}
for (i = 0; i < 5; i++)
{
DelayMs(100);
}
}
}
void Timer0_Init(void)
{
TMOD &= 0xF0;
TMOD |= 0x01; //使用模式1,16位定时器,使用"|"符号可以在使用多个定时器时不受影响
TH0 = (65536 - 18432) / 256; //重新赋值 20ms
TL0 = (65536 - 18432) % 256;
EA = 1; //总中断打开
ET0 = 1; //定时器中断打开
TR0 = 1; //定时器开关打开
}
void Timer0_Interrupt(void) interrupt 1
{
static unsigned char time_20ms = 0; //定时变量
TH0 = (65536 - 18432) / 256; //重新赋值 20ms
TL0 = (65536 - 18432) % 256;
time_20ms++;
if (time_20ms > 50)
{
dispFlag = 1;
time_20ms = 0;
}
}
void UART_Init(void)
{
SCON = 0x50; // SCON: 模式 1, 8-bit UART, 使能接收
TMOD &= 0x0F;
TMOD |= 0x20; // TMOD: timer 1, mode 2, 8-bit 重装
TH1 = 0xFD; // TH1: 重装值 9600 波特率 晶振 11.0592MHz
TR1 = 1; // TR1: timer 1 打开
EA = 1; //打开总中断
ES = 0; //关闭串口中断
}
void UART_SendByte(unsigned char dat)
{
unsigned char time_out;
time_out = 0x00;
SBUF = dat; //将数据放入SBUF中
while ((!TI) && (time_out < 100)) //检测是否发送出去
{
time_out++;
DelayUs10x(2);
} //未发送出去 进行短暂延时
TI = 0; //清除ti标志
}
void UART_SendStr(unsigned char *s, unsigned char length)
{
unsigned char NUM;
NUM = 0x00;
while (NUM < length) //发送长度对比
{
UART_SendByte(*s); //放松单字节数据
s++; //指针++
NUM++; //下一个++
}
}
//void UART_Interrupt(void) interrupt 4 //串行中断服务程序
//{
// unsigned char u_buf;
// if (RI) //判断是接收中断产生
// {
// RI = 0; //标志位清零
// u_buf = SBUF;
// }
// if (TI) //如果是发送标志位,清零
// TI = 0;
//}
设计文件: