找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 10126|回復(fù): 6
收起左側(cè)

基于51單片機(jī)酒精濃度檢測代碼+Proteus仿真圖

  [復(fù)制鏈接]
ID:454894 發(fā)表于 2019-3-1 11:30 | 顯示全部樓層 |閱讀模式
網(wǎng)上許多酒精檢測程序都有問題,為此做的新程序。

仿真原理圖如下(proteus仿真工程文件可到本帖附件中下載)
QQ截圖20190301112607.png

調(diào)節(jié)這個(gè)電位器來改變酒精的濃度

0.png

單片機(jī)源程序如下:
  1. #include <reg52.h>                 //調(diào)用單片機(jī)頭文件
  2. #define uchar unsigned char  //無符號字符型 宏定義        變量范圍0~255
  3. #define uint  unsigned int         //無符號整型 宏定義        變量范圍0~65535
  4. uchar a_a;
  5. #include <intrins.h>
  6. #define pulse0832()_nop_();_nop_();CLK=1;_nop_();_nop_();CLK=0


  7. //數(shù)碼管段選定義      0     1    2    3    4    5         6         7          8           9       
  8. uchar code smg_du[]={0x05,0xdd,0x46,0x54,0x9c,0x34,0x24,0x5d,0x04,0x14,
  9.                                            0x0c};         //斷碼

  10. //數(shù)碼管位選定義
  11. uchar code smg_we[]={0x7f,0xbf,0xdf,0xef};
  12. uchar dis_smg[]  = {0x05,0xdd,0x46,0x54,0x9c,0x34,0x24,0x5d};       
  13. uchar smg_i = 4;    //顯示數(shù)碼管的個(gè)位數(shù)

  14. sbit CS=P3^2;                //CS定義為P1口的第4位腳,連接ADC0832CS腳
  15. sbit CLK=P3^3;                //CL定義為P1口的第3位腳,連接ADC0832SCL腳
  16. sbit DI=P3^4;
  17. sbit DO=P3^4;                //DO定義為P1口的第4位腳,連接ADC0832DO腳

  18. sbit beep = P3^6;   //蜂鳴器IO口定義
  19. long dengji,s_dengji ;     //酒精等級

  20. bit flag_300ms = 1;
  21. uchar menu_1;        //菜單設(shè)計(jì)的變量

  22. #define RdCommand 0x01 //定義ISP的操作命令
  23. #define PrgCommand 0x02
  24. #define EraseCommand 0x03
  25. #define Error 1
  26. #define Ok 0
  27. #define WaitTime 0x01 //定義CPU的等待時(shí)間
  28. sfr ISP_DATA=0xe2;  //寄存器申明
  29. sfr ISP_ADDRH=0xe3;
  30. sfr ISP_ADDRL=0xe4;                                                                                         
  31. sfr ISP_CMD=0xe7;
  32. sfr ISP_TRIG=0xe6;
  33. sfr ISP_CONTR=0xe5;

  34. /* ================ 打開 ISP,IAP 功能 ================= */
  35. void ISP_IAP_enable(void)
  36. {
  37.          EA = 0;       /* 關(guān)中斷   */
  38.          ISP_CONTR = ISP_CONTR & 0x18;       /* 0001,1000 */
  39.          ISP_CONTR = ISP_CONTR | WaitTime; /* 寫入硬件延時(shí) */
  40.          ISP_CONTR = ISP_CONTR | 0x80;       /* ISPEN=1  */
  41. }
  42. /* =============== 關(guān)閉 ISP,IAP 功能 ================== */
  43. void ISP_IAP_disable(void)
  44. {
  45.          ISP_CONTR = ISP_CONTR & 0x7f; /* ISPEN = 0 */
  46.          ISP_TRIG = 0x00;
  47.          EA   =   1;   /* 開中斷 */
  48. }
  49. /* ================ 公用的觸發(fā)代碼 ==================== */
  50. void ISPgoon(void)
  51. {
  52.          ISP_IAP_enable();   /* 打開 ISP,IAP 功能 */
  53.          ISP_TRIG = 0x46;  /* 觸發(fā)ISP_IAP命令字節(jié)1 */
  54.          ISP_TRIG = 0xb9;  /* 觸發(fā)ISP_IAP命令字節(jié)2 */
  55.          _nop_();
  56. }
  57. /* ==================== 字節(jié)讀 ======================== */
  58. unsigned char byte_read(unsigned int byte_addr)
  59. {
  60.         EA = 0;
  61.          ISP_ADDRH = (unsigned char)(byte_addr >> 8);/* 地址賦值 */
  62.          ISP_ADDRL = (unsigned char)(byte_addr & 0x00ff);
  63.          ISP_CMD   = ISP_CMD & 0xf8;   /* 清除低3位  */
  64.          ISP_CMD   = ISP_CMD | RdCommand; /* 寫入讀命令 */
  65.          ISPgoon();       /* 觸發(fā)執(zhí)行  */
  66.          ISP_IAP_disable();    /* 關(guān)閉ISP,IAP功能 */
  67.          EA  = 1;
  68.          return (ISP_DATA);    /* 返回讀到的數(shù)據(jù) */
  69. }
  70. /* ================== 扇區(qū)擦除 ======================== */
  71. void SectorErase(unsigned int sector_addr)
  72. {
  73.          unsigned int iSectorAddr;
  74.          iSectorAddr = (sector_addr & 0xfe00); /* 取扇區(qū)地址 */
  75.          ISP_ADDRH = (unsigned char)(iSectorAddr >> 8);
  76.          ISP_ADDRL = 0x00;
  77.          ISP_CMD = ISP_CMD & 0xf8;   /* 清空低3位  */
  78.          ISP_CMD = ISP_CMD | EraseCommand; /* 擦除命令3  */
  79.          ISPgoon();       /* 觸發(fā)執(zhí)行  */
  80.          ISP_IAP_disable();    /* 關(guān)閉ISP,IAP功能 */
  81. }
  82. /* ==================== 字節(jié)寫 ======================== */
  83. void byte_write(unsigned int byte_addr, unsigned char original_data)
  84. {
  85.          EA  = 0;
  86.          SectorErase(byte_addr);
  87.          ISP_ADDRH = (unsigned char)(byte_addr >> 8);  /* 取地址  */
  88.          ISP_ADDRL = (unsigned char)(byte_addr & 0x00ff);
  89.          ISP_CMD  = ISP_CMD | PrgCommand;  /* 寫命令2 */
  90.          ISP_DATA = original_data;   /* 寫入數(shù)據(jù)準(zhǔn)備 */
  91.          ISPgoon();       /* 觸發(fā)執(zhí)行  */
  92.          ISP_IAP_disable();     /* 關(guān)閉IAP功能 */
  93.          EA =1;
  94. }


  95. /***********************1ms延時(shí)函數(shù)*****************************/
  96. void delay_1ms(uint q)
  97. {
  98.         uint i,j;
  99.         for(i=0;i<q;i++)
  100.                 for(j=0;j<120;j++);
  101. }

  102. /******************把數(shù)據(jù)從單片機(jī)內(nèi)部eeprom中讀出來*****************/
  103. void read_eeprom()         //讀出保存數(shù)據(jù)
  104. {
  105.         s_dengji  = byte_read(0x2001);
  106.         s_dengji <<= 8;
  107.         s_dengji  |= byte_read(0x2000);
  108.         a_a      = byte_read(0x2058);
  109. }

  110. /******************把數(shù)據(jù)保存到單片機(jī)內(nèi)部eeprom中******************/
  111. void write_eeprom()        //保存數(shù)據(jù)
  112. {
  113.         SectorErase(0x2000);
  114.         byte_write(0x2000, s_dengji % 256);
  115.         byte_write(0x2001, s_dengji / 256);
  116.         byte_write(0x2058,a_a);       
  117. }

  118. /**************開機(jī)自檢eeprom初始化*****************/
  119. void init_eeprom()         ////開始初始化保存的數(shù)據(jù)
  120. {
  121.         read_eeprom();           //讀出保存數(shù)據(jù)
  122.         if(a_a != 33)
  123.         {
  124.                 a_a = 33;
  125.                 s_dengji = 80;
  126.                 write_eeprom();                 //保存數(shù)據(jù)
  127.         }
  128. }

  129. /***********讀數(shù)模轉(zhuǎn)換數(shù)據(jù)********************************************************/       
  130. //請先了解ADC0832模數(shù)轉(zhuǎn)換的串行協(xié)議,再來讀本函數(shù),主要是對應(yīng)時(shí)序圖來理解,本函數(shù)是模擬0832的串行協(xié)議進(jìn)行的
  131.                                                 //  1  0  0 通道
  132.                                                 //  1  1  1 通道
  133. unsigned char ad0832read()
  134. {
  135.         unsigned char i=0,ch=0,ch1=0;               
  136.                 CS=0;
  137.                 DI=1;
  138.         pulse0832();                //開始
  139.                 DI=1;
  140.                 pulse0832();                //第一個(gè)上升沿       
  141.                 DI=0;
  142.                 pulse0832();
  143.                 DI=1;            //第三個(gè)下降沿
  144.                 for(i=0;i<8;i++)
  145.                 {
  146.                         pulse0832(); //開始從第四個(gè)下降沿接收數(shù)據(jù)
  147.                         ch<<=1;
  148.                         if(DO==1)
  149.                                 ch|=0x01;                                               
  150.                 }
  151.                 for(i=0;i<8;i++)
  152.                 {                        //接收校驗(yàn)數(shù)據(jù)
  153.                         ch1>>=1;
  154.                         if(DO==1)
  155.                                 ch1|=0x80;
  156.                         pulse0832();
  157.                 }
  158.                 CS=1;       
  159.                 return(ch==ch1)?ch:0;                                //與校驗(yàn)數(shù)據(jù)比較,正確就返回?cái)?shù)據(jù),否則返回0       
  160.        

  161. }

  162. /********************獨(dú)立按鍵程序*****************/
  163. uchar key_can;         //按鍵值

  164. void key()         //獨(dú)立按鍵程序
  165. {
  166.         static uchar key_new;
  167.         key_can = 20;                   //按鍵值還原
  168.         P2 |= 0x0f;                                        //把按鍵的IO口輸出為高電平
  169.         if((P2 & 0x0f) != 0x0f)                //按鍵按下
  170.         {
  171.                 delay_1ms(1);                     //按鍵消抖動
  172.                 if(((P2 & 0x0f) != 0x0f) && (key_new == 1))
  173.                 {                                                //確認(rèn)是按鍵按下
  174.                         key_new = 0;
  175.                         switch(P2 & 0x0f)
  176.                         {
  177.                                 case 0x0e: key_can = 3; break;           //得到k1鍵值
  178.                                 case 0x0d: key_can = 2; break;           //得到k2鍵值
  179.                                 case 0x0b: key_can = 1; break;           //得到k3鍵值
  180. //                                case 0x07: key_can = 4; break;           //得到k4鍵值
  181.                         }
  182.                 }                       
  183.         }
  184.         else
  185.                 key_new = 1;       
  186. }



  187. /****************按鍵處理數(shù)碼管顯示函數(shù)***************/
  188. void key_with()
  189. {
  190.         if(key_can == 1)                   //設(shè)置鍵
  191.         {
  192.                 menu_1 ++;
  193.                 if(menu_1 >= 2)
  194.                 {
  195.                         menu_1 = 0;
  196.                 }
  197.         }
  198.         if(menu_1 == 1)                        //設(shè)置酒精報(bào)警值
  199.         {
  200.                 smg_i = 4;                    //顯示4位數(shù)碼管
  201.                 if(key_can == 2)
  202.                 {
  203.                         s_dengji ++ ;                 //加1
  204.                         if(s_dengji > 500)
  205.                                 s_dengji = 500;
  206.                 }
  207.                 if(key_can == 3)
  208.                 {
  209.                         s_dengji -- ;                //減1       
  210.                         if(s_dengji <= 1)
  211.                                 s_dengji = 1;
  212.                 }
  213.                 dis_smg[0] = smg_du[s_dengji % 10];                   //取個(gè)位顯示
  214.                 dis_smg[1] = smg_du[s_dengji / 10 % 10] ;      //取十位顯示
  215.                 dis_smg[2] = smg_du[s_dengji / 100 % 10] ;           //
  216.                 dis_smg[3] = 0x05;         //a
  217.                 write_eeprom();        //保存數(shù)據(jù)
  218.         }       
  219. }  


  220. /***********************數(shù)碼顯示函數(shù)*****************************/
  221. void display()
  222. {
  223.         static uchar i;       
  224.         P1 = 0xff;                         //消隱                                           
  225.         P2 = smg_we[i];                          //位選
  226.         P1 = dis_smg[i];                 //段選          
  227.         i ++;
  228.         if(i >= smg_i)
  229.                 i = 0;         
  230. }


  231. /*************定時(shí)器0初始化程序***************/
  232. void time_init()          
  233. {
  234.         EA   = 1;                   //開總中斷
  235.         TMOD = 0X01;          //定時(shí)器0、定時(shí)器1工作方式1
  236.         ET0  = 1;                  //開定時(shí)器0中斷
  237.         TR0  = 1;                  //允許定時(shí)器0定時(shí)
  238. }



  239. /****************報(bào)警函數(shù)***************/
  240. void clock_h_l()
  241. {
  242.         static uchar value;
  243.         if((dengji >= s_dengji))                //報(bào)警
  244.         {
  245.                 value ++;
  246.                 if(value >= 2)
  247.                 {
  248.                         value = 10;
  249.                         beep = ~beep;          //蜂鳴器報(bào)警
  250.                 }
  251.         }else
  252.         {
  253.                 if((dengji < s_dengji))          //取消報(bào)警
  254.                 {
  255.                         value = 0;
  256.                         beep = 1;
  257.                 }       
  258.         }

  259. }

  260. /****************主函數(shù)***************/
  261. void main()
  262. {
  263.         beep = 0;                                //開機(jī)叫一聲   
  264.         delay_1ms(150);
  265.         P0 = P1 = P2 = P3 = 0xff;                //單片機(jī)IO口初始化為1
  266.         time_init();                                //初始化定時(shí)器
  267.         init_eeprom();              //開始初始化保存的數(shù)據(jù)
  268.         while(1)
  269.         {
  270.                 key();                                        //獨(dú)立按鍵程序
  271.                 if(key_can < 10)
  272.                 {
  273.                         key_with();                        //按鍵按下要執(zhí)行的程序
  274.                 }
  275.                 if(flag_300ms == 1)
  276.                 {               
  277.                         flag_300ms = 0;
  278.                         clock_h_l();
  279.                         dengji = ad0832read();       
  280.                         dengji = dengji * 450 / 255.0;
  281.                     dengji = dengji - 100;              //首先減去零點(diǎn)漂移,一般是100mV
  282.                         if(dengji < 0)
  283.                         {
  284.                                 dengji = 0;       
  285.                         }
  286.                        
  287.                         dengji = dengji * 2;             //將mV轉(zhuǎn)變成mg/L,系數(shù)需要校準(zhǔn)   
  288.                                                                   //電壓每升高0.1V,實(shí)際被測氣體的濃度增加20ppm
  289.                                                                   //1ppm=1mg/kg=1mg/L=1×10-6 常用來表示氣體濃度,或者溶液濃度。      
  290.                         if(menu_1 == 0)
  291.                         {
  292.                                 if(dengji >= 1000)
  293.                                         smg_i = 4;
  294.                                 else
  295.                                         smg_i = 3;
  296.                                 dis_smg[3]=smg_du[dengji/1000%10];        //千位
  297.                                 dis_smg[2]=smg_du[dengji/100%10];        //百位
  298.                                 dis_smg[1]=smg_du[dengji/10%10];        //十位
  299.                                 dis_smg[0]=smg_du[dengji%10];            //個(gè)位        ADC0832為8位ADC,數(shù)值為0~255,我們將其分開放入l_tmpdate數(shù)組中顯示
  300. ……………………

  301. …………限于本文篇幅 余下代碼請從51黑下載附件…………
復(fù)制代碼

所有資料51hei提供下載:
酒精濃度檢測單片機(jī).zip (90.71 KB, 下載次數(shù): 177)


評分

參與人數(shù) 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎勵!

查看全部評分

回復(fù)

使用道具 舉報(bào)

ID:521885 發(fā)表于 2019-4-26 12:45 | 顯示全部樓層
很有幫助
回復(fù)

使用道具 舉報(bào)

ID:555934 發(fā)表于 2019-6-5 20:40 | 顯示全部樓層
為什么運(yùn)行不了?
回復(fù)

使用道具 舉報(bào)

ID:555934 發(fā)表于 2019-6-5 21:06 | 顯示全部樓層
您好,我用了這個(gè)程序,程序調(diào)試成功,但是proteus仿真不出數(shù),最近設(shè)計(jì)很急,請問您能幫我解答一下嗎?
回復(fù)

使用道具 舉報(bào)

ID:791410 發(fā)表于 2020-6-28 01:04 來自觸屏版 | 顯示全部樓層
感謝大佬
回復(fù)

使用道具 舉報(bào)

ID:782510 發(fā)表于 2020-12-10 22:43 | 顯示全部樓層
你好,請問您能夠詳細(xì)解釋一下這個(gè)程序?qū)崿F(xiàn)的功能嗎?麻煩具體一點(diǎn),謝謝!
回復(fù)

使用道具 舉報(bào)

ID:863012 發(fā)表于 2020-12-22 19:42 | 顯示全部樓層
你好,可以具體說下各個(gè)模塊的作用以及如何實(shí)現(xiàn)的,來做個(gè)設(shè)計(jì)思路說明嗎?
回復(fù)

使用道具 舉報(bào)

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規(guī)則

小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表