【Proteus仿真】数字温度计,利用 Mega16 控制 DS18B20 ,若温度达到设定阈值,即可报警提醒(用串口控制停止报警、用键盘输入改变报警阈值)

(代码在文末)

工程文件和代码下载链接如下(求求下载前点个赞支持一下吧QAQ,博主自己做出来这个也hin累的)

链接: https://pan.baidu.com/s/1-aRZjyRZodzLcw83yx64kA  密码: og0w

一、实验目的 

设计实现一个简易数字温度计,利用 Mega16 控制 DS18B20 获取室内环境温度,并显示。 

若温度达到设定阈值,即可报警提醒。 

二、实验项目与具体要求 

(1) 实验项目:简易数字温度计。 

(2) 具体要求: 

1.液晶屏(

LCD1602/ LCD12864)或数码管显示当前的温度; 

2.利用按键设置温度上限; 

3.当温度超过上限时,进行报警。 

4.通过 PC 机“串口助手软件”向单片机发送命令字符串,控制报警蜂鸣器的关闭。 

三、实验方案设计 

1. 实验的方案的论述和分析 

①在 proteus 中搭建电路图。 

选择 mega16 单片机,液晶屏显示器,PC 机,RN1 排阻,温度传感器,蜂鸣器,三极管,电 

阻若干,按键若干,并连接硬件图。其中 PC 机的 RXD 和 TXD 接口分别接在单片机的 PD 

接口;单片机的 PA 口为输出端口直接与显示器相连,PC 口设置为输入端口与按键区相连; 

三极管基极通过电阻连接单片机 PD 口,发射极接地,集电极连接蜂鸣器,以放大电流,达 

到报警效果。 

②在 ICCV7 中建立新程序、文件,编写程序。首先对在代码中经常需要调用的命令进行简化, 

将命令语句定义为简单的语句。然后对通信端口和部分功能进行初始化,以便后续功能使用。 

然后分别对 DS18B20、LCD1602 等功能元件编写自定义函数指令,方便后续调用的逻辑清晰 

和正确性。最后将其总结在 main 函数中,使用 while 语句进行语句的循环执行。

2.硬件电路原理图(proteus 仿真图)

 

3.软件流程图、重要数据结构、重要控制参数设计等 

①软件流程图:

完整代码如下 

#include<iom16v.h>
#include<math.h>
#define uchar unsigned char
#define uint unsigned int
#define rs_h (PORTB|=0x01)
#define rs_l (PORTB&=0xfe)
#define rs_o (DDRB|=0x01)
#define rw_h (PORTB|=0x02)
#define rw_l (PORTB&=0xfd)
#define rw_o (DDRB|=0x02)
#define en_h (PORTB|=0x04)
#define en_l (PORTB&=0xfb)
#define en_o (DDRB|=0x04)
#define temp_h (PORTB|=0x08)
#define temp_l (PORTB&=0xf7)
#define LCD PORTA
#define tmp (PINB&0x08)
#define temp_o (DDRB|=0x08)
#define temp_i (DDRB&=0xf7)
uchar dat1,dat2;//保存读出的温度
#define led_o (DDRB|=0x10)
#define led_l (PORTB&=0xef)
#define led_h (PORTB|=0x10)

unsigned char f=0;
unsigned char c=0;
int threshold=25;
unsigned char cah=0;
int date;

#include <macros.h>
unsigned char Rev_data=0;
#pragma interrupt_handler uart_Rev_int:iv_USART_RXC//串口接收中断
void uart_Rev_int(void)
{
    Rev_data = UDR;
    while(!Rev_data);
    if (Rev_data == 0xA1)
    {
        f=1;
    }
    if(Rev_data == 0xA2)
    {
        f=0;
    }
    while(!(UCSRA & (1<<UDRE)));
    UDR = Rev_data+0x10;
    Rev_data = 0;
}

uchar chg=0;
void delayms();
#pragma interrupt_handler int0_isr:2//外部中断 CHANGE按钮
void int0_isr(void)
{
    delayms(200);
    cah=1;
}

void initcom(void)//初始化端口
{
    DDRD = 0x0A;
    PORTD=0x04;
    GICR |=(1<<INT0);
    SREG|=0x80;
    UBRRL = 25;
    UCSRC = 0x86;
    UCSRB = 0x98;           //接收发送控制位,打开串口10011000
    SEI();
    DDRC=0x0f;
    PORTC=0x00;

}


unsigned long int dat=0;
uchar flag=0;
void delayms(uint z)	  //8M晶振下,延时1ms
{
    uint x,y;
    for(x=z; x>0; x--)
        for(y=1333; y>0; y--);
}

void Ds18b20_reset(void)//DS18B20温度传感器初始化
{
    uint count;
    uint i=60000;
    temp_o;
    temp_l;
    for(count=700; count>0; count--); //延时480us
    temp_h;
    temp_i;//不须配置PORT内部上拉电阻,MCU输入输出自动切换
    while((tmp==0x08));//&&(i>0)) i--;
    led_o;
    led_l;//开指示灯
    for(count=700; count>0; count--); //延时480us
}
void Ds18b20_write(uchar dat)//向DS18B20写一个字节
{
    uchar count;
    uchar i;
    temp_o;
    for(i=8; i>0; i--)
    {
        temp_l;
        for(count=2; count>0; count--);
        //temp_h;//不能有此语句
        if(dat&0x01==0x01)
            temp_h;
        else
            temp_l;
        for(count=120; count>0; count--); //延时60us
        temp_h;
        dat>>=1;
    }
}
uchar Ds18b20_read(void)//从DS18B20读一个字节
{
    uchar i,datt;
    uchar count;
    for(i=8; i>0; i--)
    {
        datt>>=1;
        temp_o;
        temp_l;
        for(count=2; count>0; count--);
        temp_h;//此语句必须有,参考datasheet的P15
        for(count=1; count>0; count--);
        temp_i;
        if(tmp==0x08)
            datt|=0x80;
        for(count=120; count>0; count--);
    }
    return datt;
}
void lcd_com(uchar com)//向LCD1602写命令
{
    rs_o;
    rw_o;
    en_o;
    DDRA=0xff;
    rs_l;
    rw_l;
    LCD=com;
    delayms(1);
    en_h;
    delayms(1);
    en_l;
}
void lcd_dat(uchar dat)//向LCD1602写数据
{
    rs_o;
    rw_o;
    DDRA=0xff;
    en_o;
    rs_h;
    rw_l;
    LCD=dat;
    delayms(1);
    en_h;
    delayms(1);
    en_l;
}
void lcd_write(uchar c,uchar r,uchar dat)//向LCD1602指定行、指定列、写数据
{
    lcd_com(0x80+0x40*c+r);
    lcd_dat(dat);
    delayms(1);
}
void lcd_init(void)//LCD1602初始化,初始化后第一行显示temperature:,第二行显示.C
{
    DDRA=0xff;
    DDRB|=0x17;
    lcd_com(0x38);
    lcd_com(0x0c);
    lcd_com(0x06);
    lcd_write(0,2,0x54);
    lcd_write(0,3,0x65);
    lcd_write(0,4,0x6d);
    lcd_write(0,5,0x70);
    lcd_write(0,6,0x65);
    lcd_write(0,7,0x72);
    lcd_write(0,8,0x61);
    lcd_write(0,9,0x74);
    lcd_write(0,10,0x75);
    lcd_write(0,11,0x72);
    lcd_write(0,12,0x65);
    lcd_write(0,13,0x3a);
    lcd_write(1,11,0xdf);
    lcd_write(1,12,0x43);
}
void show(void)//把温度值送LCD1602显示
{
    if(dat2>=240)//遗留问题,温度为25时读出dat1=144,dat2=1正确,但却进入if(dat2&0xf8==0xf8)分支;
    {
        dat=(~(dat2*256+dat1)+1)*(0.0625*10);//取反加一,保留一位小数
        flag=1;
    }
    else
    {
        dat=(dat2*256+dat1)*(0.0625*10);
        flag=0;
    };

    if(flag==1)//负温度显示
    {
        lcd_write(1,10,0x30+dat%10);
        lcd_write(1,9,0xa5);
        lcd_write(1,8,0x30+dat%100/10);
        lcd_write(1,7,0x30+dat%1000/100);
        lcd_write(1,6,0x30+dat/1000);
        lcd_write(1,5,0x2d);
    }
    if(flag==0)//正温度显示
    {
        lcd_write(1,10,0x30+dat%10);
        lcd_write(1,9,0xa5);
        lcd_write(1,8,0x30+dat%100/10);
        lcd_write(1,7,0x30+dat%1000/100);
        lcd_write(1,6,0x30+dat/1000);
        lcd_write(1,5,0x20);//显示空格,把之前显示屏上可能存在的负号替换成空格
    }
}

unsigned int Num_key[3][3]= {{1,2,3},{4,5,6},{7,8,9}};

int scan(void)//用扫描的方式读取一个键盘上的输入
{
    while(1)
    {
        uint i,j,ff=0;//I用于确定列,j用于确定行
        uchar ii=1,jj=1; //ii和jj分别用于和PORTC做与运算,改变或取得PORTC 0〜3位和PORTC 4〜7位中需要位置的电平信息
        if((PINC&0x80)!=0)
        {
            delayms(5);
            if((PINC&0x80)!=0)//如果按下的是“ENTER”按钮
            {
                while((PINC&0x80)!=0);
                return -3;
            }
        }
        for(i=0; i<3; i++)
        {
            PORTC=ii;
            jj=1;
            for(j=0; j<3; j++)
            {
                if((PINC&(jj*16))!=0)
                {
                    delayms(5);//延时等待电压稳定,防抖
                    if((PINC&(jj*16))!=0)
                    {
                        while((PINC&(jj*16))!=0);
                        return Num_key[j][I];返回该位置按钮对应的数值
                    }
                }
                jj*=2;//指向下一个引脚
                delayms(5);
            }
            if((PINC&(jj*16))!=0)
            {
                delayms(5);
                if((PINC&(jj*16))!=0)
                {
                    if(i==1)//如果按下的是0
                    {
                        while((PINC&(jj*16))!=0);
                        return 0;
                    }
                    else if(i==0)//如果按下的是负号“-”
                    {
                        while((PINC&(jj*16))!=0);
                        return -1;
                    }
                }
            }
            delayms(5);
            PORTC=0x00;
            ii*=2;
        }

    }
}

void main(void)
{
    lcd_init();
    initcom();
    while(1)
    {
        dat1=0x00;
        dat2=0x00;
        Ds18b20_reset();//温度传感器初始化
        Ds18b20_write(0xcc);
        Ds18b20_write(0x44);//发送温度转换命令
        delayms(1000);//延时1s,等待温度转换完成
        Ds18b20_reset();
        Ds18b20_write(0xcc);
        Ds18b20_write(0xbe);//发送读温度寄存器命令
        dat1=Ds18b20_read();
        dat2=Ds18b20_read();
        show();
        if(cah)
        {
            int sum=0,symb=1;
            delayms(200);
            while(1)
            {
                int inp=scan();
                if(inp==-3)//如果按下ENTER键,代表温度输入完成
                {
                    threshold=sum*symb;
                    break;
                }
                else if(inp==-1)//如果按下负号键,则符号symb赋值为-1
                {
                    symb=-1;
                }
                else//若输入其他数字,则将原油总额*10+当前获取的数字
                {
                    sum=sum*10+inp;
                }
            }
            cah=0;
            PORTC=0;
        }
        if(flag)//如果是负数
        {
            date=-dat;
        }
        else//如果是正数
        {
            date=dat;
        }
        if(date>=(threshold*10)&&f==0&&c==0)//threshold是报警阈值,date是当前温度*10,因为要有一位小数,而date是整型变量,因此*10后成为整数保存。f代表的是串口是否发送指令关闭了报警功能。c代表此时是否正在报警。
        {
            c=1;
            PORTD+=8;
        }
        else if((date<(threshold*10)||f==1)&&c==1)
        {
            c=0;
            PORTD-=8;
        }
        led_h;//关指示灯
        delayms(1000);
    }
}

//串口通信速率我记得好像是18200
//时钟为内部的8M晶振
//在按键盘进行阈值更改时,务必按键时间长一点(比如1s),不然可能检测不到

  • 18
    点赞
  • 34
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Bill-Besti

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值