久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 8389|回復: 6
收起左側

酒精傳感器電路原理圖與單片機代碼仿真分享

  [復制鏈接]
ID:433919 發表于 2018-11-30 13:25 | 顯示全部樓層 |閱讀模式
制作出來的實物圖如下:
0.png

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

調節這個電位器來改變酒精的濃度

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


  6. //數碼管段選定義      0     1    2    3    4    5         6         7          8           9       
  7. uchar code smg_du[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x04,0x14,
  8.                                            0x0c,0xa4,0x27,0xc4,0x26,0x2e,0xff};         //斷碼

  9. //數碼管位選定義
  10. uchar code smg_we[]={0x7f,0xbf,0xdf,0xef};
  11. uchar dis_smg[8]  = {0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8};       
  12. uchar smg_i = 4;    //顯示數碼管的個位數

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

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

  18. bit flag_300ms = 1;
  19. uchar menu_1;        //菜單設計的變量

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

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


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

  100. /******************把數據從單片機內部eeprom中讀出來*****************/
  101. void read_eeprom()         //讀出保存數據
  102. {
  103.         s_dengji  = byte_read(0x2001);
  104.         s_dengji <<= 8;
  105.         s_dengji  |= byte_read(0x2000);
  106.         a_a      = byte_read(0x2058);
  107. }

  108. /******************把數據保存到單片機內部eeprom中******************/
  109. void write_eeprom()        //保存數據
  110. {
  111.         SectorErase(0x2000);
  112.         byte_write(0x2000, s_dengji % 256);
  113.         byte_write(0x2001, s_dengji / 256);
  114.         byte_write(0x2058,a_a);       
  115. }

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

  127. /***********讀數模轉換數據********************************************************/       
  128. //請先了解ADC0832模數轉換的串行協議,再來讀本函數,主要是對應時序圖來理解,本函數是模擬0832的串行協議進行的
  129.                                                 //  1  0  0 通道
  130.                                                 //  1  1  1 通道
  131. unsigned char ad0832read(bit SGL,bit ODD)
  132. {
  133.         unsigned char i=0,value=0,value1=0;               
  134.                 SCL=0;
  135.                 DO=1;
  136.                 CS=0;                //開始
  137.                 SCL=0;                //第一個上升沿       
  138.                 SCL=1;
  139.                 DO=SGL;
  140.                 SCL=1;          //第二個上升沿
  141.                 SCL=0;
  142.                 DO=ODD;
  143.                 SCL=1;            //第三個上升沿
  144.                 SCL=0;            //第三個下降沿
  145.                 DO=1;
  146.                 for(i=0;i<8;i++)
  147.                 {
  148.                         SCL=0;
  149.                         SCL=1; //開始從第四個下降沿接收數據
  150.                         value<<=1;
  151.                         if(DO)
  152.                                 value++;                                               
  153.                 }
  154.                 for(i=0;i<8;i++)
  155.                 {                        //接收校驗數據
  156.                         value1>>=1;
  157.                         if(DO)
  158.                                 value1+=0x80;
  159.                         SCL=0;
  160.                         SCL=1;
  161.                 }
  162.                 CS=1;
  163.                 SCL=1;       
  164.                 if(value==value1)                                //與校驗數據比較,正確就返回數據,否則返回0       
  165.                         return value;
  166.         return 0;
  167. }

  168. /********************獨立按鍵程序*****************/
  169. uchar key_can;         //按鍵值

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



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


  226. /***********************數碼顯示函數*****************************/
  227. void display()
  228. {
  229.         static uchar i;       
  230.         P1 = 0xff;                         //消隱                                           
  231.         P2 = smg_we[i];                          //位選
  232.         P1 = dis_smg[i];                 //段選          
  233.         i ++;
  234.         if(i >= smg_i)
  235.                 i = 0;         
  236. }


  237. /*************定時器0初始化程序***************/
  238. void time_init()          
  239. {
  240.         EA   = 1;                   //開總中斷
  241.         TMOD = 0X01;          //定時器0、定時器1工作方式1
  242.         ET0  = 1;                  //開定時器0中斷
  243.         TR0  = 1;                  //允許定時器0定時
  244. }



  245. /****************報警函數***************/
  246. void clock_h_l()
  247. {
  248.         static uchar value;
  249.         if((dengji >= s_dengji))                //報警
  250.         {
  251.                 value ++;
  252.                 if(value >= 2)
  253.                 {
  254.                         value = 10;
  255.                         beep = ~beep;          //蜂鳴器報警
  256.                 }
  257.         }else
  258.         {
  259.                 if((dengji < s_dengji))          //取消報警
  260.                 {
  261.                         value = 0;
  262.                         beep = 1;
  263.                 }       
  264.         }

  265. }

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

  307. /*************定時器0中斷服務程序***************/
  308. void time0_int() interrupt 1
  309. {       
  310.         static uchar value;
  311.         TH0 = 0xf8;
  312.         TL0 = 0x30;     // 2ms
  313.         display();
  314.         value ++;         
  315.         if(value % 150 == 0)
  316.         {
  317.                 flag_300ms = 1;           //300ms
  318.                 value = 0;
  319.         }

  320. }

復制代碼

所有資料51hei提供下載:
51_úì.rar (244.6 KB, 下載次數: 122)

評分

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

查看全部評分

回復

使用道具 舉報

ID:298003 發表于 2018-12-5 04:20 | 顯示全部樓層
下載來看看  嘿嘿嘿
回復

使用道具 舉報

ID:636964 發表于 2019-12-12 09:59 來自手機 | 顯示全部樓層
有沒有hex文件的c文件啊
回復

使用道具 舉報

ID:636964 發表于 2019-12-12 09:59 來自手機 | 顯示全部樓層
下載下來  沒有
回復

使用道具 舉報

ID:711443 發表于 2020-3-25 12:22 來自手機 | 顯示全部樓層
請問酒精傳感器怎么制作在原理圖上?
回復

使用道具 舉報

ID:867995 發表于 2020-12-25 20:51 | 顯示全部樓層
真的是對的嗎?  圖片感覺挺模糊的
回復

使用道具 舉報

ID:1076325 發表于 2023-11-3 16:44 | 顯示全部樓層
117299 發表于 2020-12-25 20:51
真的是對的嗎?  圖片感覺挺模糊的

請問這個原理圖能做出實物嗎
回復

使用道具 舉報

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

本版積分規則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 亚洲一区二区三区四区五区中文 | 久久久久一区二区 | 久久久成人精品 | 日韩国产欧美视频 | 成人一区二区三区在线观看 | 国产在线一级片 | 免费视频一区二区 | 国产成人免费视频网站高清观看视频 | 91精品国产自产精品男人的天堂 | 国产精品免费观看视频 | 国产一区视频在线 | 中国大陆高清aⅴ毛片 | 操到爽 | 综合色影院 | 一区二区视频在线 | 欧美日韩亚洲系列 | 超碰在线人人干 | 精品av| 一区二区三区 在线 | 狠狠干av| 亚洲视频二区 | 在线看片福利 | 免费av一区二区三区 | www国产亚洲精品 | 天天干天天干 | 四虎影院新地址 | 天天拍夜夜爽 | 天堂av资源 | 精品一区二区三区不卡 | 亚洲国产免费 | 国产成人精品在线播放 | 日韩精品一二三 | 人人人干 | 精品日本中文字幕 | 欧美中文在线 | 精品欧美一区免费观看α√ | 一级片毛片 | 国产亚洲成av人片在线观看桃 | 中文字幕一二三 | 天堂男人av | 亚洲视频欧美视频 |