欧美极品高清xxxxhd,国产日产欧美最新,无码AV国产东京热AV无码,国产精品人与动性XXX,国产传媒亚洲综合一区二区,四库影院永久国产精品,毛片免费免费高清视频,福利所导航夜趣136

標題: 雙通道電壓采集電路設計 完整的單片機課程設計下載帶程序 [打印本頁]

作者: 塵歸土    時間: 2016-6-5 11:22
標題: 雙通道電壓采集電路設計 完整的單片機課程設計下載帶程序
附件里面有設計任務,根據附件原理圖改正里面的程序。完整的word文檔請在本文最后下載附件,下面是部分內容預覽:




目     錄
一、原理 ……………………………………………………………………………………………1
二、硬件介紹 ………………………………………………………………………………………1
   2.1 單片機介紹…………………………………………………………………………………1
2.2 AT24C02I2C數據存儲 ……………………………………………………………………2
   2.3 PCF8591 A/D轉換 …………………………………………………………………………2
   2.4 LCD1602 液晶顯示模塊……………………………………………………………………3
   硬件連接及工作過程簡介………………………………………………………………………4
   3.1 硬件連接……………………………………………………………………………………4
   3.2 工作過程簡介………………………………………………………………………………5
三、工作流程圖 ……………………………………………………………………………………6
四、程序清單 ………………………………………………………………………………………7
五、PCB圖 …………………………………………………………………………………………18
六、總結……………………………………………………………………………………………18
八、參考文獻………………………………………………………………………………………19


一、原理本文所設計的是一種以單片機STC12C5A60S2(引腳排序及基本功能同AT89S51)作為主控芯片的雙通道電壓采集電路,將采集到的模擬信號電壓值利用PCF8591來進行模數轉換,PCF8591是一個單片集成、單獨供電、低功耗、8-bitCMOS數據獲取信號。其中PCF8591共有4個模擬輸入、一個模擬輸出和一個串行口I2C總線接口,通過PLCF8591將轉換成的數字信號電壓值傳給液晶顯示屏,并將所測電壓值在LCD1602液晶顯示屏中顯示出來,而且要利用AT24C04設置電壓上下限,在電壓超出上下限時顯示屏要有所表示.本文主要針對雙通道的數據采集與顯示,繼而推廣多通道數據采集的方法,重點分析介紹了基于AT89S51數字電壓采集的硬件設計和軟件設計。


圖3-1硬件連接原理圖




SDA、SCL是IIC數據線和時鐘線,可連接于單片機任何IO管腳在。在本次設計中通過芯片AT24C02來編寫所測電壓的上下限值。
我們這次試驗所測得電壓是直流電壓0至5伏是模擬信號由于我們所測得電壓要在 LCD1602 液晶顯示屏中顯示出來,但是LCD1602液晶顯示屏接受和表達數字信號,所以要用PCF8591 A/D轉換芯片把采集的模擬信號電壓值轉換成數值信號,經過AT89S51再把信號傳給LCD液晶顯示屏,最終在液晶顯示屏上顯示出電壓值。如果采集的電壓值在設定的電壓值之間,則電壓值能在液晶顯示屏上顯示出電壓,如果超出了則在液晶顯示屏上有所表示。在此次試驗中因為是雙通道采集電壓,所以我們測兩條電路的電壓,我在這兩條電路上各串聯了一個滑動變阻器,用來改變所測的電壓值。

五、程序清單
  1. #include "common.h"

  2. float v_low = 2.8;  //電壓范圍下限
  3. float v_high = 4.5;  //電壓范圍上限
  4. uchar BUF[2];  //從at24c04中讀取的數據存儲在這
  5. uchar w_buf[2]; //寫入at24c04中的數據存在這里

  6. //延時函數大約1ms
  7. void delay_ms(int ms)
  8. {
  9.         int i;
  10.         while(ms--)
  11.         {
  12.         for(i=0;i<100;i++);
  13.         }
  14. }

  15. //連續寫入兩個數據
  16. void AT24C04_Write(uchar dat_1,uchar dat_2)
  17. {
  18.         AT24C04_Start(); //起始信號
  19.         AT24C04_SendByte(0xa0); //發送設備地址+寫信號
  20.         AT24C04_SendByte(0x00); //發送存儲單元地址
  21.         AT24C04_SendByte(dat_1); //寫入數據1
  22.         AT24C04_SendByte(dat_2); //寫入數據2
  23.         AT24C04_Stop(); //停止信號
  24. }

  25. //連續讀出兩個數據
  26. void AT24C04_ReadPage()
  27. {
  28. uchar i;
  29. AT24C04_Start(); //起始信號
  30. AT24C04_SendByte(0xa0); //發送設備地址+寫信號
  31. AT24C04_SendByte(0x00); //發送存儲單元地址
  32. AT24C04_Start(); //起始信號
  33. AT24C04_SendByte(0xa1); //發送設備地址+讀信號
  34. for (i=0; i<2; i++)
  35. {
  36. BUF[i] = AT24C04_RecvByte();
  37. if (i == 1)
  38. {
  39. AT24C04_SendACK(1); //最后一個數據需要會NAK
  40. }
  41. else
  42. {
  43. AT24C04_SendACK(0); //回應ACK
  44. }
  45. }
  46. AT24C04_Stop(); //停止信號
  47. }

  48. //LCD顯示數字處理函數
  49. void show_num(uchar num,uchar flag)
  50. {
  51.         uchar temp[3],i;
  52.         float re_val;  //電壓實際值
  53.         float read_low,read_h; //將從AT24C04中讀取電壓范圍數據轉化成對應小數形式

  54.         re_val = num/255.0*5.0; //實際電壓計算公式

  55.         num = re_val*10; //保留一位小數

  56.          AT24C04_ReadPage();//從AT24C04中讀取電壓范圍數據,存儲在BUF數組中
  57.                                           // BUF[0]表示下限,BUF[1]表示上限
  58.         //存入AT24C04中的數據是uchar unsigned型,在這里轉化成float型
  59.         read_low =         BUF[0]/10.0;
  60.         read_h  = BUF[1]/10.0;
  61.         if(re_val >= read_low && re_val <= read_h) //如果電壓在范圍內
  62.            {
  63.                 temp[0] = num/10 + '0';
  64.                 temp[1] = '.';
  65.                 temp[2] = num%10 + '0';
  66.                 }else if(re_val < read_low ) //如果低于設定值
  67.            {
  68.                    temp[0] = 'L';
  69.                 temp[1] = 'O';
  70.                 temp[2] = 'W';           
  71.            }else if(re_val > read_h ) //如果高于設定值
  72.            {
  73.                    temp[0] = 'H';
  74.                 temp[1] = 'I';
  75.                 temp[2] = 'G';         
  76.                 }
  77.         
  78.   for(i=0;i<3;i++)                 //用液晶顯示結果
  79.   {
  80.            LCD_Manifest(flag,3+i,temp[i]);
  81.   }

  82.         //顯示電壓范圍的上限
  83.     num = read_h * 10;
  84.         temp[0] = num/10 + '0';
  85.         temp[1] = '.';
  86.         temp[2] = num%10 + '0';
  87.   for(i=0;i<3;i++)                 //用液晶顯示結果
  88.   {
  89.           LCD_Manifest(1,12+i,temp[i]);
  90.   }

  91.    //顯示電壓范圍的下限
  92.     num = read_low * 10;
  93.         temp[0] = num/10 + '0';
  94.         temp[1] = '.';
  95.         temp[2] = num%10 + '0';
  96.   for(i=0;i<3;i++)                 //用液晶顯示結果
  97.   {
  98.           LCD_Manifest(2,12+i,temp[i]);
  99.   }
  100. }

  101. //主函數
  102. void main()
  103. {
  104.         uchar val;        //電壓值

  105.         LCD_Init();          //液晶初始化
  106.         IIC_Init();         //I2C總線初始化

  107.          w_buf[0] =         v_low*10;
  108.         w_buf[1] =         v_high*10;
  109.         AT24C04_Write(w_buf[0],w_buf[1]); //將數據寫入AT24C04

  110.         LCD_Manifest(1,0,'V');        //在lcd1602中顯示字符
  111.         LCD_Manifest(1,1,'1');
  112.         LCD_Manifest(1,2,':');
  113.         LCD_Manifest(2,0,'V');
  114.         LCD_Manifest(2,1,'2');
  115.         LCD_Manifest(2,2,':');

  116.         LCD_Manifest(1,10,'H');
  117.         LCD_Manifest(1,11,':');
  118.         LCD_Manifest(2,10,'L');
  119.         LCD_Manifest(2,11,':');


  120.         while(1)
  121.         {
  122.                 //按鍵檢測程序
  123.                 if(key_1 == 0)           //調節范圍的下限
  124.                 {
  125.                    delay_ms(100); //延時消除抖動
  126.                    if(key_1 == 0)
  127.                    {
  128.                   //         v_low = v_low - 0.1;
  129.                         // if(v_low < 0.0)
  130.                          //  v_low = v_high;

  131.                         w_buf[0] =         v_low*10;
  132.                         w_buf[1] =         v_high*10;
  133.                         AT24C04_Write(w_buf[0],w_buf[1]); //將數據寫入AT24C04
  134.                    }
  135.                 }
  136.                 if(key_2 == 0)                //調節范圍的上限
  137.                 {
  138.                    delay_ms(100); //延時消除抖動
  139.                    if(key_2 == 0)
  140.                    {
  141.                             v_high = v_high + 0.1;
  142.                          if(v_high > 5.0)
  143.                            v_high = v_low;

  144.                         w_buf[0] =         v_low*10;
  145.                         w_buf[1] =         v_high*10;
  146.                         AT24C04_Write(w_buf[0],w_buf[1]); //將數據寫入AT24C04
  147.                    }
  148.                 }


  149.            ADC_Send_Byte(0x90,0x41);//通道一ADC轉換         
  150.            val=ADC_Receive_Byte(0x90);        //讀取AD轉換的值
  151.            show_num(val,1);
  152.            ADC_Send_Byte(0x90,0x42);//通道二ADC轉換         
  153.            val=ADC_Receive_Byte(0x90);        //讀取AD轉換的值
  154.            show_num(val,2);


  155.         }
  156. }

  157. #include<iic.h>
  158. void IIC_Init()//總線初始化
  159. {
  160.   
  161.   SCL=1;
  162.   IIC_DELAY();
  163.   SDA=1;
  164.   IIC_DELAY();
  165. }
  166. void IIC_Start() //啟動信號
  167. {

  168.   SDA=1;
  169.   IIC_DELAY();
  170.   SCL=1;
  171.   IIC_DELAY();
  172.   SDA=0;
  173.   IIC_DELAY();
  174. }
  175. void IIC_Stop()        //停止信號
  176. {
  177.   
  178.    SDA=0;
  179.    IIC_DELAY();
  180.    SCL=1;
  181.    IIC_DELAY();
  182.    SDA=1;
  183.    IIC_DELAY();
  184. }
  185. void IIC_Ack() //應答信號
  186. {
  187.    unsigned char i;
  188.    SCL=1;
  189.    IIC_DELAY();;
  190.    while((SDA==1)&&(i<255))i++;
  191.    SCL=0;
  192.    IIC_DELAY();
  193. }
  194. void IIC_Write_Byte(u8 date)//寫一個字節
  195. {
  196.    u8 i,temp;
  197.    temp=date;
  198.    for(i=0;i<8;i++)
  199.    {
  200.              temp=temp<<1;
  201.           SCL=0;
  202.           IIC_DELAY();
  203.           SDA=CY;//最高位將移入PSW寄存器的CY位中,然后將CY賦給SDA
  204.           IIC_DELAY();
  205.           SCL=1;
  206.           IIC_DELAY();
  207.    }
  208.    SCL=0;
  209.    IIC_DELAY();
  210.    SDA=1;
  211.    IIC_DELAY();
  212. }
  213. u8 IIC_Read_Byte()//讀一個字節
  214. {
  215.    u8 i,temp;
  216.    for(i=0;i<8;i++)
  217.    {
  218.           SCL=0;
  219.           IIC_DELAY();
  220.           temp=(temp<<1)|SDA;
  221.           IIC_DELAY();
  222.           SCL=1;
  223.           IIC_DELAY();
  224.    }
  225.    SCL=0;
  226.    IIC_DELAY();
  227.    SDA=1;
  228.    IIC_DELAY();
  229.    return temp;
  230. }
  231. #include <reg51.h>
  232. #include"lcd1602.h"
  233. void delay(u8 z)//延時函數
  234. {
  235.   unsigned char x,y;
  236.   for(x=z;x>0;x--)
  237.     for(y=110;y>0;y--);
  238. }


  239. void LCD_Write_Cmd(u8 com )//液晶寫命令
  240. {
  241.    LCD_RS=0;//選擇寫命令模式
  242.    LCD_dat=com;//將要寫的命令字送到數據總線上
  243.    delay(1);
  244.    LCD_E=1;//使能端給一高脈沖,高脈沖有效
  245.    delay(1);
  246.    LCD_E=0;
  247. }
  248. void LCD_Write_Data(u8 date)//數據寫命令
  249. {
  250.    LCD_RS=1;//選擇寫數據模式
  251.    LCD_dat=date;
  252.    delay(1);
  253.    LCD_E=1;//使能端給一高脈沖,高脈沖有效
  254.    delay(1);
  255.    LCD_E=0;
  256. }
  257. void LCD_Init()//液晶初始化
  258. {
  259.   
  260.    LCD_WR=0;
  261.    LCD_Write_Cmd(0x38);//設置16*2顯示,5*7點陣,8位數據接口
  262.    LCD_Write_Cmd(0x0c);//設置開顯示,不顯示光標
  263.    LCD_Write_Cmd(0x06);//寫一個字符后地址指針自動加1

  264.    LCD_Write_Cmd(0x80);//數據指針定位到第一行第一個字處

  265. }
  266. // 使得寫命令和寫數據通過一個函數完成
  267. void LCD_Manifest(u8 row,u8 add,u8 date)
  268. {
  269.          if(row==1)          LCD_Write_Cmd(0x80+add);
  270.         if(row==2)         LCD_Write_Cmd(0xc0+add);
  271.         LCD_Write_Data(date);
  272. }
  273. #include"iic.h"
  274. #include"pcf8591.h"

  275. //ADC發送字節[命令]數據函數
  276. void ADC_Send_Byte(u8 addr,u8 com)
  277. {
  278.    IIC_Start();              //啟動總線
  279.    IIC_Write_Byte(addr);           //發送器件地址
  280.    IIC_Ack();
  281.    IIC_Write_Byte(com);             //發送數據
  282.    IIC_Ack();
  283.    IIC_Stop();              //結束總線
  284. }

  285. //ADC讀字節數據函數
  286. u8 ADC_Receive_Byte(u8 addr)
  287. {
  288.    u8 c;
  289.    IIC_Start();          //啟動總線
  290.    IIC_Write_Byte(addr+1);      //發送器件地址,如果是讀則最低位為1即0x91
  291.    IIC_Ack();
  292.    c=IIC_Read_Byte();          //讀取數據

  293.    IIC_Ack();          //發送非就答位
  294.    IIC_Stop();           //結束總線
  295.    return(c);
  296. }
  297. #include"AT24C04.h"
  298. void Delay5us()
  299. {
  300. BYTE n = 4;
  301. while (n--)
  302. {
  303. _nop_();
  304. _nop_();
  305. }
  306. }
  307. void Delay5ms()
  308. {
  309. WORD n = 2500;
  310. while (n--)
  311. {
  312. _nop_();
  313. _nop_();
  314. _nop_();
  315. _nop_();
  316. _nop_();
  317. }
  318. }
  319. void AT24C04_Start()
  320. {
  321. AT_SDA = 1; //拉高數據線
  322. AT_SCL = 1; //拉高時鐘線
  323. Delay5us(); //延時
  324. AT_SDA = 0; //產生下降沿
  325. Delay5us(); //延時
  326. AT_SCL = 0; //拉低時鐘線
  327. }
  328. void AT24C04_Stop()
  329. {
  330. AT_SDA = 0; //拉低數據線
  331. AT_SCL = 1; //拉高時鐘線
  332. Delay5us(); //延時
  333. AT_SDA = 1; //產生上升沿
  334. Delay5us(); //延時
  335. }
  336. void AT24C04_SendACK(bit ack)
  337. {
  338. AT_SDA = ack; //寫應答信號
  339. AT_SCL = 1; //拉高時鐘線
  340. Delay5us(); //延時
  341. AT_SCL = 0; //拉低時鐘線
  342. Delay5us(); //延時
  343. }
  344. bit AT24C04_RecvACK()
  345. {
  346. AT_SCL = 1; //拉高時鐘線
  347. Delay5us(); //延時
  348. CY = AT_SDA; //讀應答信號
  349. AT_SCL = 0; //拉低時鐘線
  350. Delay5us(); //延時
  351. return CY;
  352. }
  353. void AT24C04_SendByte(BYTE dat)
  354. {
  355. BYTE i;
  356. for (i=0; i<8; i++) //8位計數器
  357. {
  358. dat <<= 1; //移出數據的最高位
  359. AT_SDA = CY; //送數據口
  360. AT_SCL = 1; //拉高時鐘線
  361. Delay5us(); //延時
  362. AT_SCL = 0; //拉低時鐘線
  363. Delay5us(); //延時
  364. }
  365. AT24C04_RecvACK();
  366. }
  367. BYTE AT24C04_RecvByte()
  368. {
  369. BYTE i;
  370. BYTE dat = 0;
  371. AT_SDA = 1; //使能內部上拉,準備讀取數據
  372. for (i=0; i<8; i++) //8位計數器
  373. {
  374. dat <<= 1;
  375. AT_SCL = 1; //拉高時鐘線
  376. Delay5us(); //延時
  377. dat |= AT_SDA; //讀數據
  378. AT_SCL = 0; //拉低時鐘線
  379. Delay5us(); //延時
  380. }
  381. return dat;
  382. }

復制代碼

七、總結
本次課程設計,我學到了很多的東西,對匯編語言程序設計加深了理解和掌握,我們這次設計的要求主要是熟悉微型計算機技術中所學到的幾中芯片的使用,掌握它們的初始化編程,并學會他們的基本應用,這是一個比較全面的設計,和以前的實驗相比較復雜很多,一方面要求我們設計硬件電路,另一方面需要對設計的硬件電路進行編程,這就需要軟件和硬件相結合,具體的電路具體分析,在此次設計中還用到了AT24C02芯片,在此次之前我們從沒用到這個芯片,對這個芯片非常陌生,而且網上也沒有什么系統的解釋,和AT89S51的管腳的連線也不清楚,但是最后通過不停地調試,最后還是被解決了。通過這次課程設計后,我對學硬件有了一定的體會,它要求我們認真仔細的做好每個環節,對所有的問題要考慮周全,它和軟件不同,對具體的電路要編寫適合該電路的程序,電路一改,程序也要跟著改,從中我也學會了很多實際應用的知識。通過本次設計,我才發現我所學的東西的膚淺,知識面的不牢固,在選擇方案的時候,產生模棱兩可的情況。經常是邊設計邊復習學習過的課本知識。這些使我知道“紙上談兵”是絕對不行,只有通過實踐,自己親自動手試一試,才能發現自己知識的缺陷,才能更好的與所學的知識相結合。在課程設計中我們每個人都能學到很多,有很大的收獲。做任何事情所要有的態度和心態,首先我明白了做學問要一絲不茍,對于出現的任何問題和偏差都不要輕視,要通過正確的途徑去解決,在做事情的過程中要有耐心和毅力,不要一遇到困難就打退堂鼓,只要堅持下去就可以找到思路去解決問題的。在工作中要學會與人合作的態度,認真聽取別人的意見,這樣做起事情來就可以事倍功半。

課程設計任務書下載: 任務書10----雙通道電壓采集電路設計(LCD顯示).doc (36.5 KB, 下載次數: 19)
完整的設計文檔下載: 單片機課設.doc (214.5 KB, 下載次數: 49)





歡迎光臨 (http://www.raoushi.com/bbs/) Powered by Discuz! X3.1