登录|立即注册|使用QQ帐号登录
论坛 > 24小时必答区
发帖|
看1662|回2|收藏
楼主 ID:905971 只看他
2021-5-5 22:10
仿真图如图所示
1620223725(1).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. #define pulse0832()_nop_();_nop_();CLK=1;_nop_();_nop_();CLK=0


  7. //数码管段选定义      0     1    2    3    4    5         6         7          8           9        
  8. uchar code smg_du[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90};         //断码

  9. //数码管位选定义
  10. uchar code smg_we[]={0x7f,0xbf,0xdf,0xef};
  11. uchar dis_smg[]  = {0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8};        
  12. uchar smg_i = 4;    //显示数码管的个位数

  13. sbit CS=P3^2;                //CS定义为P1口的第4位脚,连接ADC0832CS脚
  14. sbit CLK=P3^3;                //CL定义为P1口的第3位脚,连接ADC0832SCL脚
  15. sbit DI=P3^4;
  16. sbit DO=P3^4;   //DO定义为P1口的第4位脚,连接ADC0832DO脚
  17. sbit beep = P3^6;   //蜂鸣器IO口定义
  18. long dengji,s_dengji ;     //酒精等级

  19. bit flag_300ms = 1;
  20. uchar menu_1;        //菜单设计的变量

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

  33. /* ================ 打开 ISP,IAP 功能 ================= */
  34. void ISP_IAP_enable(void)
  35. {
  36.          EA = 0;       /* 关中断   */
  37.          ISP_CONTR = ISP_CONTR & 0x18;       /* 0001,1000 */
  38.          ISP_CONTR = ISP_CONTR | WaitTime; /* 写入硬件延时 */
  39.          ISP_CONTR = ISP_CONTR | 0x80;       /* ISPEN=1  */
  40. }
  41. /* =============== 关闭 ISP,IAP 功能 ================== */
  42. void ISP_IAP_disable(void)
  43. {
  44.          ISP_CONTR = ISP_CONTR & 0x7f; /* ISPEN = 0 */
  45.          ISP_TRIG = 0x00;
  46.          EA   =   1;   /* 开中断 */
  47. }
  48. /* ================ 公用的触发代码 ==================== */
  49. void ISPgoon(void)
  50. {
  51.          ISP_IAP_enable();   /* 打开 ISP,IAP 功能 */
  52.          ISP_TRIG = 0x46;  /* 触发ISP_IAP命令字节1 */
  53.          ISP_TRIG = 0xb9;  /* 触发ISP_IAP命令字节2 */
  54.          _nop_();
  55. }
  56. /* ==================== 字节读 ======================== */
  57. unsigned char byte_read(unsigned int byte_addr)
  58. {
  59.         EA = 0;
  60.          ISP_ADDRH = (unsigned char)(byte_addr >> 8);/* 地址赋值 */
  61.          ISP_ADDRL = (unsigned char)(byte_addr & 0x00ff);
  62.          ISP_CMD   = ISP_CMD & 0xf8;   /* 清除低3位  */
  63.          ISP_CMD   = ISP_CMD | RdCommand; /* 写入读命令 */
  64.          ISPgoon();       /* 触发执行  */
  65.          ISP_IAP_disable();    /* 关闭ISP,IAP功能 */
  66.          EA  = 1;
  67.          return (ISP_DATA);    /* 返回读到的数据 */
  68. }
  69. /* ================== 扇区擦除 ======================== */
  70. void SectorErase(unsigned int sector_addr)
  71. {
  72.          unsigned int iSectorAddr;
  73.          iSectorAddr = (sector_addr & 0xfe00); /* 取扇区地址 */
  74.          ISP_ADDRH = (unsigned char)(iSectorAddr >> 8);
  75.          ISP_ADDRL = 0x00;
  76.          ISP_CMD = ISP_CMD & 0xf8;   /* 清空低3位  */
  77.          ISP_CMD = ISP_CMD | EraseCommand; /* 擦除命令3  */
  78.          ISPgoon();       /* 触发执行  */
  79.          ISP_IAP_disable();    /* 关闭ISP,IAP功能 */
  80. }
  81. /* ==================== 字节写 ======================== */
  82. void byte_write(unsigned int byte_addr, unsigned char original_data)
  83. {
  84.          EA  = 0;
  85.          SectorErase(byte_addr);
  86.          ISP_ADDRH = (unsigned char)(byte_addr >> 8);  /* 取地址  */
  87.          ISP_ADDRL = (unsigned char)(byte_addr & 0x00ff);
  88.          ISP_CMD  = ISP_CMD | PrgCommand;  /* 写命令2 */
  89.          ISP_DATA = original_data;   /* 写入数据准备 */
  90.          ISPgoon();       /* 触发执行  */
  91.          ISP_IAP_disable();     /* 关闭IAP功能 */
  92.          EA =1;
  93. }


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

  101. /******************把数据从单片机内部eeprom中读出来*****************/
  102. void read_eeprom()         //读出保存数据
  103. {
  104.         s_dengji  = byte_read(0x2001);
  105.         s_dengji <<= 8;
  106.         s_dengji  |= byte_read(0x2000);
  107.         a_a      = byte_read(0x2058);
  108. }

  109. /******************把数据保存到单片机内部eeprom中******************/
  110. void write_eeprom()        //保存数据
  111. {
  112.         SectorErase(0x2000);
  113.         byte_write(0x2000, s_dengji % 256);
  114.         byte_write(0x2001, s_dengji / 256);
  115.         byte_write(0x2058,a_a);        
  116. }

  117. /**************开机自检eeprom初始化*****************/
  118. void init_eeprom()         ////开始初始化保存的数据
  119. {
  120.         read_eeprom();           //读出保存数据
  121.         if(a_a != 33)
  122.         {
  123.                 a_a = 33;
  124.                 s_dengji = 125;
  125.                 write_eeprom();                 //保存数据
  126.         }
  127. }

  128. /***********读数模转换数据********************************************************/        
  129. //请先了解ADC0832模数转换的串行协议,再来读本函数,主要是对应时序图来理解,本函数是模拟0832的串行协议进行的
  130.                                                 //  1  0  0 通道
  131.                                                 //  1  1  1 通道
  132. unsigned char ad0832read()
  133. {
  134.         unsigned char i=0,ch=0,ch1=0;               
  135.                 CS=0;
  136.                 DI=1;
  137.         pulse0832();                //开始
  138.                 DI=1;
  139.                 pulse0832();                //第一个上升沿        
  140.                 DI=0;
  141.                 pulse0832();
  142.                 DI=1;            //第三个下降沿
  143.                 for(i=0;i<8;i++)
  144.                 {
  145.                         pulse0832(); //开始从第四个下降沿接收数据
  146.                         ch<<=1;
  147.                         if(DO==1)
  148.                                 ch|=0x01;                                                
  149.                 }
  150.                 for(i=0;i<8;i++)
  151.                 {                        //接收校验数据
  152.                         ch1>>=1;
  153.                         if(DO==1)
  154.                                 ch1|=0x80;
  155.                         pulse0832();
  156.                 }
  157.                 CS=1;        
  158.                 return(ch==ch1)?ch:0;                                //与校验数据比较,正确就返回数据,否则返回0        
  159.         

  160. }

  161. /********************独立按键程序*****************/
  162. uchar key_can;         //按键值

  163. void key()         //独立按键程序
  164. {
  165.         static uchar key_new;
  166.         key_can = 20;                   //按键值还原
  167.         P2 |= 0x0f;                                        //把按键的IO口输出为高电平
  168.         if((P2 & 0x0f) != 0x0f)                //按键按下
  169.         {
  170.                 delay_1ms(1);                     //按键消抖动
  171.                 if(((P2 & 0x0f) != 0x0f) && (key_new == 1))
  172.                 {                                                //确认是按键按下
  173.                         key_new = 0;
  174.                         switch(P2 & 0x0f)
  175.                         {
  176.                                 case 0x0e: key_can = 3; break;           //得到k1键值
  177.                                 case 0x0d: key_can = 2; break;           //得到k2键值
  178.                                 case 0x0b: key_can = 1; break;           //得到k3键值
  179. //                                case 0x07: key_can = 4; break;           //得到k4键值
  180.                         }
  181.                 }                        
  182.         }
  183.         else
  184.                 key_new = 1;        
  185. }



  186. /****************按键处理数码管显示函数***************/
  187. void key_with()
  188. {
  189.         if(key_can == 1)                   //设置键
  190.         {
  191.                 menu_1 ++;
  192.                 if(menu_1 >= 2)
  193.                 {
  194.                         menu_1 = 0;
  195.                 }
  196.         }
  197.         if(menu_1 == 1)                        //设置酒精报警值
  198.         {
  199.                 smg_i = 4;                    //显示4位数码管
  200.                 if(key_can == 2)
  201.                 {
  202.                         s_dengji ++ ;                 //加1
  203.                         if(s_dengji > 500)
  204.                                 s_dengji = 500;
  205.                 }
  206.                 if(key_can == 3)
  207.                 {
  208.                         s_dengji -- ;                //减1        
  209.                         if(s_dengji <= 1)
  210.                                 s_dengji = 1;
  211.                 }
  212.                 dis_smg[0] = smg_du[s_dengji % 10];                   //取个位显示
  213.                 dis_smg[1] = smg_du[s_dengji / 10 % 10] ;      //取十位显示
  214.                 dis_smg[2] = smg_du[s_dengji / 100 % 10] ;           //
  215.                 dis_smg[3] = 0xc0;         //a
  216.                 write_eeprom();        //保存数据
  217.         }        
  218. }  


  219. /***********************数码显示函数*****************************/
  220. void display()
  221. {
  222.         static uchar i;        
  223.         P1 = 0xff;                         //消隐                                          
  224.         P2 = smg_we[i];                          //位选
  225.         P1 = dis_smg[i];                 //段选           
  226.         i ++;
  227.         if(i >= smg_i)
  228.                 i = 0;         
  229. }


  230. /*************定时器0初始化程序***************/
  231. void time_init()         
  232. {
  233.         EA   = 1;                   //开总中断
  234.         TMOD = 0X01;          //定时器0、定时器1工作方式1
  235.         ET0  = 1;                  //开定时器0中断
  236.         TR0  = 1;                  //允许定时器0定时
  237. }



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

  258. }

  259. /****************主函数***************/
  260. void main()
  261. {
  262.         beep = 0;                                //开机叫一声   
  263.         delay_1ms(150);
  264.         P0 = P1 = P2 = P3 = 0xff;                //单片机IO口初始化为1
  265.         time_init();                                //初始化定时器
  266.         init_eeprom();              //开始初始化保存的数据
  267.         while(1)
  268.         {
  269.                 key();                                        //独立按键程序
  270.                 if(key_can < 10)
  271.                 {
  272.                         key_with();                        //按键按下要执行的程序
  273.                 }
  274.                 if(flag_300ms == 1)
  275.                 {               
  276.                         flag_300ms = 0;
  277.                         clock_h_l();
  278.                         dengji = ad0832read();        
  279.                         dengji = dengji * 450 / 255.0;
  280.                     dengji = dengji - 100;              //首先减去零点漂移,一般是100mV
  281.                         if(dengji < 0)
  282.                         {
  283.                                 dengji = 0;        
  284.                         }
  285.                         
  286.                         dengji = dengji * 2;             //将mV转变成mg/L,系数需要校准   
  287.                                                                   //电压每升高0.1V,实际被测气体的浓度增加20ppm
  288.                                                                   //1ppm=1mg/kg=1mg/L=1×10-6 常用来表示气体浓度,或者溶液浓度。      
  289.                         if(menu_1 == 0)
  290.                         {
  291.                                 if(dengji >= 1000)
  292.                                         smg_i = 4;
  293.                                 else
  294.                                         smg_i = 3;
  295.                                 dis_smg[3]=smg_du[dengji/1000%10];        //千位
  296.                                 dis_smg[2]=smg_du[dengji/100%10];        //百位
  297.                                 dis_smg[1]=smg_du[dengji/10%10];        //十位
  298.                                 dis_smg[0]=smg_du[dengji%10];            //个位        ADC0832为8位ADC,数值为0~255,我们将其分开放入l_tmpdate数组中显示
  299.                         }
  300.                 }
  301.         }
  302. }

  303. /*************定时器0中断服务程序***************/
  304. void time0_int() interrupt 1
  305. {        
  306.         static uchar value;
  307.         TH0 = 0xf8;
  308.         TL0 = 0x30;     // 2ms
  309.         display();
  310.         value ++;         
  311.         if(value % 150 == 0)
  312.         {
  313.                 flag_300ms = 1;           //300ms
  314.                 value = 0;                                                                                          
  315.         }

  316. }
求大神告知 本人还是初学者 很多地方都还不了解
沙发 ID:957243 只看他
2021-8-13 21:20
请问电压到浓度是怎样换算的
板凳 ID:957243 只看他
2021-8-13 21:21
请问电压到浓度的怎么换算的

51黑电子论坛

Powered by Discuz! X3.1

首页|标准版|触屏版|电脑版