仿真原理圖如下(proteus仿真工程文件可到本帖附件中下載)
51hei.gif (179.67 KB, 下載次數: 50)
下載附件
2022-4-9 18:16 上傳
單片機源程序如下:
- #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)
2022-4-9 18:18 上傳
點擊文件名下載附件
下載積分: 黑幣 -5
|