找回密码
 注册

QQ登录

只需一步,快速开始

搜索

51单片机ADC0832能换量程数字电压表程序源码和proteus仿真

[复制链接]
eng 发表于 2021-6-14 02:42:23 | 显示全部楼层 |阅读模式
仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)
1.png

部分源码:完整源码请下载附件
  1. /*数字电压表,能在0-5V,0-50V,0-250V之间自动切换量程*/
  2. #include<reg51.h>
  3. #define uchar unsigned char
  4. #define uint unsigned int
  5. #include <intrins.h>
  6. code uchar TAB[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90}; //定义数码管显示数值
  7. uchar str_TME[4]={0,0,0,0,}; //定义数码管显示初始值
  8. unsigned char data1;     //data1用来存放转换以后的数据
  9. sbit dang1=P2^7;
  10. sbit dang2=P2^6;
  11. sbit key1=P0^0;
  12. sbit led1=P1^7;//0-50V指示
  13. sbit led2=P1^6;//0-250V批示
  14. sbit led3=P1^5;//0-5V指示
  15. uchar flag;//用来作量程标志
  16. void delay(uchar ms) //定义ms延迟程序
  17. {
  18. uchar i;
  19. while(ms--)
  20. for(i=0;i<125;i++);
  21. }
  22. uchar tmel=0;
  23. bit sim=1;//转换标志
  24. /***********************计数器中断函数,用于控制电压转*********************
  25. *******************时间间隔,此程序设定间隔为1S电压转换一次***************/
  26. void  tme_tr0(void)  interrupt 1
  27. {
  28. TL0=0xb0;
  29. TH0=0x3c;
  30. if(++tmel==20)
  31.     {
  32.         tmel=0;
  33.         sim=1;//1S时间到,实现一次电压转换
  34.     }
  35. }
  36. /*****************定义数码管显示(共阳数码管)***************************/
  37. sbit k1=P1^0;   //第一位
  38. sbit k2=P1^1;   //第二位
  39. sbit k3=P1^2;   //第三位
  40. sbit k4=P1^3;   //第四位
  41. void VAL_xs0()//0-250v显示
  42. {
  43. k1=1; k2=1; k3=1; k4=1;
  44. P3=TAB[str_TME[0]];   
  45. k1=0;  //第一位显示
  46. delay(5);
  47. k1=1;
  48. P3=TAB[str_TME[1]];
  49. k2=0; //第二位显示
  50. delay(5);
  51. k2=1;
  52. P3=(TAB[str_TME[2]])&0x7f;//显示小数点
  53. k3=0;  //第三位显示
  54. delay(5);
  55. k3=1;
  56. P3=TAB[str_TME[3]];
  57. k4=0;  //第四位显示
  58. delay(5);
  59. k4=1;
  60. }
  61. void VAL_xs1()//0-50v显示
  62. {
  63. k1=1; k2=1; k3=1; k4=1;
  64. P3=TAB[str_TME[0]];   
  65. k1=0;  //第一位显示
  66. delay(5);
  67. k1=1;
  68. P3=(TAB[str_TME[1]])&0x7f;//显示小数点
  69. k2=0; //第二位显示
  70. delay(5);
  71. k2=1;
  72. P3=TAB[str_TME[2]];
  73. k3=0;  //第三位显示
  74. delay(5);
  75. k3=1;
  76. P3=TAB[str_TME[3]];
  77. k4=0;  //第四位显示
  78. delay(5);
  79. k4=1;
  80. }
  81. void VAL_xs2()//0-5v显示
  82. {
  83. k1=1; k2=1; k3=1; k4=1;
  84. P3=(TAB[str_TME[0]])&0x7f;   
  85. k1=0;  //第一位显示
  86. delay(5);
  87. k1=1;
  88. P3=TAB[str_TME[1]];//显示小数点
  89. k2=0; //第二位显示
  90. delay(5);
  91. k2=1;
  92. P3=TAB[str_TME[2]];
  93. k3=0;  //第三位显示
  94. delay(5);
  95. k3=1;
  96. P3=TAB[str_TME[3]];
  97. k4=0;  //第四位显示
  98. delay(5);
  99. k4=1;
  100. }

  101. /******************************************************************************
  102.                             AD0832AD转换子程序
  103. *******************************************************************************/
  104. sbit bADcs=P2^2;   //片选位
  105. sbit bADcl=P2^1;   //时钟位
  106. sbit bADda=P2^0;   //数据位
  107. void ad(void)
  108. {           uchar i;
  109.      bADcs = 0;//当ADC0832未工作时其CS输入端应为高电平,此时芯片禁用,开始工作CS为低电平
  110.          bADcl=0;  //第一个时钟下降沿前da为1,第二个与第三时钟下降沿前的数据为通道选择
  111.          bADda=1;  //选置起始位
  112.          bADcl=1;
  113.          bADcl=0;   //   1down
  114.          bADda=1;   //通道选择第1位
  115.          bADcl=1;
  116.          bADcl=0;        //   2 down
  117.          bADda=0;   //通道选择第2位,通道选择为1,0选通道0
  118.          bADcl=1;
  119.          bADcl=0;        //   3 down
  120.          bADda=1;
  121.          bADcl=1;
  122.          bADcl=0;        //   4 down
  123.          for(i=8;i>0;i--)
  124.                  {               
  125.                  data1<<=1;  //从第7位开始,要左移
  126.                  bADcl=0;
  127.                 bADcl=1;
  128.                 if(bADda==1) data1|=0x01;  //如果输出1,data1最后一位补1
  129.                  }
  130.         bADcs=1;        //转换完后CS置1
  131. }
  132. void changs()  //转换程序
  133. {
  134.     double sum;         
  135.     uchar val_Integer;             //定义整数变量
  136.     unsigned int val_Decimal;      //定义小数变量
  137.    if((dang1==0)&&(dang2==1)) //档位转换(0-100V)
  138.     { flag=0;
  139.       sum=data1*0.0196078*20 ; //扩展50倍
  140.       led1=1;led2=0;led3=1;
  141.       val_Integer=(uchar)sum;
  142.       val_Decimal=(unsigned int)((sum-val_Integer)*10);
  143.       str_TME[3]=val_Decimal;
  144.       str_TME[2]=val_Integer%10;
  145.       str_TME[1]=val_Integer/10%10;
  146.       str_TME[0]=val_Integer/100;
  147.       if((sum<=25)&&(sum>=5))//如果在25V量程,则切换成25V量程
  148.         {
  149.          dang1=1; dang2=0;
  150.         }
  151.        if(sum<5)//小数字时切换成5V量程
  152.         { dang1=0;dang2=0;}   
  153.     }
  154.      else if((dang1==1)&(dang2==0))//档位转换(0-25V?
  155.         { flag=1;
  156.       sum=data1*0.0196078*5 ; //扩展5倍
  157.       led2=1;led1=0;led3=1;
  158.       val_Integer=(uchar)sum;
  159.       val_Decimal=(unsigned int)((sum-val_Integer)*100);
  160.       str_TME[3]=val_Decimal%10;
  161.       str_TME[2]=val_Decimal/10;
  162.       str_TME[1]=val_Integer%10;
  163.       str_TME[0]=val_Integer/10;
  164.       if(sum>24.9)//大于25V切换成大量程
  165.         {
  166.            dang1=0; dang2=1;
  167.         }
  168.        if(sum<5)//小于5V切换成5V量程
  169.         { dang1=0;dang2=0;}   
  170.     }
  171.      else if((dang1==0)&&(dang2==0))//档位转换(0-5V?
  172.           {  flag=2;
  173.         sum=data1*0.0196078 ;
  174.         led2=1;led1=1;  led3=0;
  175.         val_Integer=(uchar)sum;
  176.         val_Decimal=(unsigned int)((sum-val_Integer)*1000);
  177.         str_TME[3]=val_Decimal%10;
  178.         str_TME[2]=val_Decimal/10%10;
  179.         str_TME[1]=val_Decimal/100;
  180.         str_TME[0]=val_Integer;
  181.       
  182.         if(sum>4.9)//大于5V切换成25V量程
  183.           { dang1=1;dang2=0;
  184.               if(sum>25)//大于50V切换成250V量程
  185.               {
  186.                  dang1=0; dang2=1;
  187.               }
  188.           }
  189.       }
  190. }
  191. /*****************************************************
  192.                        主程序
  193. *******************************************************/
  194. main()
  195. {   
  196.     P1=0xc0;
  197.     IE=0x82;//开中断
  198.     TMOD=0x01;//定时器设置
  199.     IP=0x01;
  200.     TL0=0xb0;//初值
  201.     TH0=0x3c;
  202.     TR0=1;//启动
  203.    dang1=0;dang2=1;
  204. while(1)
  205.     {  
  206.        switch(flag)//用flag标志不同量程
  207.        {
  208.         case 0:
  209.         VAL_xs0();//250量程显示
  210.         break;
  211.         case 1:
  212.         VAL_xs1();//50量程显示
  213.         break;
  214.         case 2:
  215.         VAL_xs2();//5V量程显示
  216.         break;
  217.         }            
  218.         if(sim==1)//1秒到后,实现一次电压转换
  219.             {         
  220.             ad();   //电压转换
  221.             changs();       //数据转换
  222.             sim=0;
  223.             }        
  224.     }
  225. }
复制代码
完整源码和proteus文件: 数显稳压电源设计.zip (372.54 KB, 售价: 5 E币)
您需要登录后才可以回帖 登录 | 注册

本版积分规则

QQ|手机版|小黑屋|ELEOK |网站地图

GMT+8, 2024-4-26 08:19

Powered by Discuz! X3.5

© 2001-2024 Discuz! Team.

快速回复 返回顶部 返回列表