仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)
单片机源程序如下:
- #include <reg51.h>
- unsigned char code seg7code[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,
- 0x07,0x7f,0x6f};
- code unsigned char seg7codeB[]={0xbf,0x86,0xdb,0xcf,0xe6,0xed,
- 0xfd,0x87,0xff,0xef,0x00}; //带小数点的共阴数码管段码
- sbit DQ=P3^5; //数据传输线接单片机的相应的引脚
- sbit motor = P3^0;
- sbit up = P3^1;
- sbit down = P3^2;
- unsigned char count0 = 50;
- unsigned char count1 = 0;
- unsigned char PWM = 50;
-
- unsigned char tempL=0; //设全局变量
- unsigned char tempH=0;
- unsigned int sdata; //测量到的温度的整数部分
- unsigned char xiaoshu1; //小数第一位
- unsigned char xiaoshu2; //小数第二位
- unsigned char xiaoshu; //两位小数
- bit fg=1; //温度正负标志
- void delay(unsigned char i)
- {
- for(i;i>0;i--);
- }
- void delay1ms(int n)
- {
- unsigned char i;
- for(i=124*n;i>0;i--); //延时124*8+10=1002us
- }
- void Init_DS18B20(void)
- {
- unsigned char x=0;
- DQ=1; //DQ先置高
- delay(8); //稍延时
- DQ=0; //发送复位脉冲
- delay(80); //延时(>480us)
- DQ=1; //拉高数据线
- delay(5); //等待(15~60us)
- x=DQ; //用X的值来判断初始化有没有成功,18B20存在的话X=0,否则X=1
- delay(20);
- }
- ReadOneChar(void) //主机数据线先从高拉至低电平1us以上,再使数据线升为高电平,从而产生读信号
- {
- unsigned char i=0; //每个读周期最短的持续时间为60us,各个读周期之间必须有1us以上的高电平恢复期
- unsigned char dat=0;
- for (i=8;i>0;i--) //一个字节有8位
- {
- DQ=1;
- delay(1);
- DQ=0;
- dat>>=1; //数据移位
- DQ=1;
- if(DQ)
- dat|=0x80; //得读取数据线得到一个状态
- delay(4);
- }
- return(dat);
- }
- void WriteOneChar(unsigned char dat)
- {
- unsigned char i=0; //数据线从高电平拉至低电平,产生写起始信号。15us之内将所需写的位送到数据线上,
- for(i=8;i>0;i--) //在15~60us之间对数据线进行采样,如果是高电平就写1,低写0发生。
- {
- DQ=0; //在开始另一个写周期前必须有1us以上的高电平恢复期。
- DQ=dat&0x01; //按低位发送数据
- delay(5);
- DQ=1;
- dat>>=1;
- }
- delay(4);
- }
- void ReadTemperature(void)
- {
- Init_DS18B20(); //初始化
- WriteOneChar(0xcc); //跳过读序列号的操作
- WriteOneChar(0x44); //启动温度转换
- delay(125); //转换需要一点时间,延时
- Init_DS18B20(); //初始化
- WriteOneChar(0xcc); //跳过读序列号的操作
- WriteOneChar(0xbe); //读温度寄存器(头两个值分别为温度的低位和高位)
- tempL=ReadOneChar(); //读出温度的低位LSB
- tempH=ReadOneChar(); //读出温度的高位MSB
-
- if(tempH>0x7f) //最高位为1时温度是负
- {
- tempL=~tempL; //补码转换,取反加一
- tempH=~tempH+1;
- fg=0; //读取温度为负时fg=0
- }
- sdata = tempL/16+tempH*16; //整数部分
- xiaoshu1 = (tempL&0x0f)*10/16; //小数第一位
- xiaoshu2 = (tempL&0x0f)*100/16%10;//小数第二位
- xiaoshu=xiaoshu1*10+xiaoshu2; //小数两位
- }
- void Led(unsigned int date)
- {
- P1=0xfb; //P1.0=0,选通第一位
- P0=seg7code[date/10]; //十位数,查表,输出
- delay1ms(1);
- P1=0xfd; //P1.1=0,选通第二位,个位数
- P0=seg7codeB[date%10];
- delay1ms(1);
-
- P1=0xfe; //P1.2=0,选通第三位,小数点第一位
- P0=seg7code[xiaoshu1];
- delay1ms(1);
- P1=0xf7; //P1.3=0,选通第四位
- P0=0x58;
- delay1ms(1);
- }
- void main()
- {
- unsigned int i;
- while(1)
- {
- ReadTemperature();
- for(i=0;i<20;i++)
- {
- Led(sdata);
- }
- }
- }
- void Timer0_int(void) interrupt 1
- {
- TR0 = 0;
- TL0 = (65536-1000) % 256;
- TH0 = (65536-1000) / 256 ;
- TR0 = 1;
- if(++count1 < count0)
- {
- PWM = 1;
- }
- else
- PWM = 0;
- if(count1 >= 100)
- {
- count1=0;
- }
- }
复制代码
Keil代码与Proteus仿真2个文件下载:
仿真代码.7z
(676.96 KB, 下载次数: 66)
|