制作出來的實物圖如下:
仿真原理圖如下(proteus仿真工程文件可到本帖附件中下載)
上位機:
安裝程序解壓到同一目錄下。再運行setup安裝。。。。
溫度計.exe
編寫環境Win10 x64. VS2010 x86 C++ MFC
測試環境Win10 x64.
在32位系統上,能不能正常使用還不清楚。
所以公布了源代碼文件,如果在32位系統上,不能正常使用。
請使用源代碼文件,在32位系統上重新生成一下應用程序。。。
單片機源程序如下:
- #include <reg52.h>
- #include "math.h"
- bit flag1s = 0, _up = 0; //1s定時標志
- unsigned char T0RH = 0; //T0重載值的高字節
- unsigned char T0RL = 0; //T0重載值的低字節
- void ConfigTimer0(unsigned int ms);
- unsigned char IntToString(unsigned char *str, int dat);
- extern bit Start18B20();
- extern bit Get18B20Temp(int *temp);
- extern void InitLcd1602();
- extern void LcdShowStr(unsigned char x, unsigned char y, unsigned char *str);
- //串口初始化函數
- void InitUART()
- {
- //IP = 0x10;
- TMOD &= 0x0F;
- TMOD |= 0x20;
- SCON = 0x50;
- TH1 = 0xF3;
- TL1 = TH1;
- PCON = 0x00;
- EA = 1;
- ES = 1;
- TR1 = 1;
- }
- //串口發送1字節數據
- void SendOneByte(unsigned char c)
- {
- SBUF = c;
- while(!TI);
- TI = 0;
- }
- void main()
- {
- bit res;
- int temp, former=0xffff; //讀取到的當前溫度值
- int intT, decT; //溫度值的整數和小數部分
- unsigned char len;
- unsigned char str[12];
-
- IP = 0x10;
- InitUART();
- // EA = 1; //開總中斷
- ConfigTimer0(10); //T0定時10ms
- Start18B20(); //啟動DS18B20
- InitLcd1602(); //初始化液晶
-
- while (1)
- {
- if (flag1s || _up) //每秒更新一次溫度
- {
- flag1s = 0;
- res = Get18B20Temp(&temp); //讀取當前溫度
- if (res && ((abs(temp-former) >= 4)||_up)) //讀取成功時,刷新當前溫度顯示
- {
- former = temp; //記錄新的溫度
- _up = 0;
- SendOneByte((unsigned char) (temp/256));
- SendOneByte((unsigned char) (temp%256));
- intT = temp >> 4; //分離出溫度值整數部分
- decT = temp & 0xF; //分離出溫度值小數部分
- len = IntToString(str, intT); //整數部分轉換為字符串
- str[len++] = '.'; //添加小數點
- decT = (int)((decT*100) * 0.0625 + 0.5); //二進制的小數部分轉換為1位十進制位
- str[len++] = decT/10 + '0'; //十進制小數位再轉換為ASCII字符
- str[len++] = decT%10 + '0';
- str[len++] = 'C';
- while (len < 7) //用空格補齊到6個字符長度
- {
- str[len++] = ' ';
- }
- str[len] = '\0'; //添加字符串結束符
- LcdShowStr(0, 0, str); //顯示到液晶屏上
- }
- else //讀取失敗時,提示錯誤信息
- {
- if(!res)
- LcdShowStr(0, 0, "error!");
- }
- Start18B20(); //重新啟動下一次轉換
- }
- }
- }
- /* 整型數轉換為字符串,str-字符串指針,dat-待轉換數,返回值-字符串長度 */
- unsigned char IntToString(unsigned char *str, int dat)
- {
- signed char i = 0;
- unsigned char len = 0;
- unsigned char buf[6];
-
- if (dat < 0) //如果為負數,首先取絕對值,并在指針上添加負號
- {
- dat = -dat;
- *str++ = '-';
- len++;
- }
- do { //先轉換為低位在前的十進制數組
- buf[i++] = dat % 10;
- dat /= 10;
- } while (dat > 0);
- len += i; //i最后的值就是有效字符的個數
- while (i-- > 0) //將數組值轉換為ASCII碼反向拷貝到接收指針上
- {
- *str++ = buf[i] + '0';
- }
- *str = '\0'; //添加字符串結束符
-
- return len; //返回字符串長度
- }
- /* 配置并啟動T0,ms-T0定時時間 */
- void ConfigTimer0(unsigned int ms)
- {
- unsigned long tmp; //臨時變量
-
- tmp = 12000000 / 12; //定時器計數頻率
- tmp = (tmp * ms) / 1000; //計算所需的計數值
- tmp = 65536 - tmp; //計算定時器重載值
- tmp = tmp + 15; //補償中斷響應延時造成的誤差
- T0RH = (unsigned char)(tmp>>8); //定時器重載值拆分為高低字節
- T0RL = (unsigned char)tmp;
- TMOD &= 0xF0; //清零T0的控制位
- TMOD |= 0x01; //配置T0為模式1
- TH0 = T0RH; //加載T0重載值
- TL0 = T0RL;
- ET0 = 1; //使能T0中斷
- TR0 = 1; //啟動T0
- }
- /* T0中斷服務函數,完成1秒定時 */
- void InterruptTimer0() interrupt 1
- {
- static unsigned char tmr1s = 0;
-
- TH0 = T0RH; //重新加載重載值
- TL0 = T0RL;
- tmr1s++;
- if (tmr1s >= 100) //定時1s
- {
- tmr1s = 0;
- if(!flag1s)
- flag1s = 1;
- }
- }
- //串口中斷服務函數,
- void UARTInterrupt(void) interrupt 4
- {
- char ch=0;
- if(RI)
- {
- RI = 0;
- ch = SBUF;
- if((ch=='1') && !_up)
- _up = 1;
- //add your code here!
- }/*
- else
- TI = 0;*/
- }
復制代碼
所有資料51hei提供下載:
溫度計。單片機 仿真.rar
(106.64 KB, 下載次數: 55)
2020-11-21 00:39 上傳
點擊文件名下載附件
溫度計。上位機.rar
(130.59 KB, 下載次數: 46)
2020-11-21 01:08 上傳
點擊文件名下載附件
安裝程序.7z
(13.13 MB, 下載次數: 36)
2020-11-21 01:51 上傳
點擊文件名下載附件
|