找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 3839|回复: 3
收起左侧

单片机数字电流电压表程序Proteus仿真

[复制链接]
ID:848043 发表于 2020-11-24 20:50 | 显示全部楼层 |阅读模式
仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)
51hei.png 51hei.png

单片机源程序如下:
  1. /******引脚说明
  2. LCD液晶RS2.5  RW2.6    E2.7       DB0~DB7:P0.0~P0.7
  3. 电压测量P1.1;需要0~30v需要加分流电阻,,分辨率0.03v;
  4. 电流侧量P1.7;需要加放大电路如lm358;op07等,分辨率3ma
  5. **********/
  6. //stc12c5a60
  7. //时间20141107                           
  8. #include<STC12C5A60S2.h>
  9. #include<stdio.h>
  10. #include<intrins.h>//头文件

  11. #define uchar unsigned char//宏定义
  12. #define uint unsigned int//宏定义
  13. #define ulint unsigned long int//宏定义
  14. #define N 21//ADC中值滤波次数,必须为基数
  15. #define ISP_TRIG() ISP_TRIG=0x5A,ISP_TRIG=0xA5//ISP触发命令

  16. sbit rs=P3^5;//液晶RS
  17. sbit rw=P3^6;//液晶RW
  18. sbit e=P3^4;//液晶E

  19. void AD_init(void);//ADC初始化
  20. void delay(uint z);//延时函数
  21. uint AD_get(uchar haha );//读ADC的数值
  22. void LCD_data(uchar shuju);//写数据
  23. void zhiling(uchar zhilin);//写指令
  24. void LCD_init(void);//初始化LCD
  25. void tablex(uint tab);//数据分解
  26. uint filter(uchar dat);//中值滤波
  27. void timer_init(void);//定时器初始化

  28. uchar h1,h2,h3,h4;//LCD显示变量
  29. ulint mas,maz;//容量计算

  30. uchar table[]="0123456789";//显示数组
  31. uchar table1[]=".VAWa";

  32. void main()
  33. {
  34.         ulint tempv,tempa;//存放电压电流值
  35.         AD_init();//ADC初始化
  36.         timer_init();//定时器初始化
  37.         LCD_init();//LCD初始化
  38.         while(1)
  39.         {
  40.                 tempv=filter(0x88);//设置ADC通道0、开始转换
  41.                 tablex(tempv*3);//分解数据
  42.                 zhiling(0x80);//LCD地址
  43.                 LCD_data(table[h1]);//
  44.                 LCD_data(table[h2]);//
  45.                 LCD_data(table1[0]);//
  46.                 LCD_data(table[h3]);//
  47.                 LCD_data(table[h4]);//
  48.                 LCD_data(table1[1]);//显示
  49.                
  50.                 tempa=filter(0x8f);//设置ADC通道7、开始转换
  51.                 if(tempa<=10)//防止数据成负数
  52.                         tempa=10;
  53.                 mas=tempa*3-30;//计算容量取样赋值
  54.                 tablex(tempa*3-30);//减去LM358失调的数值
  55.                 zhiling(0x88);//LCD地址
  56.                 LCD_data(table[h1]);
  57.                 LCD_data(table1[0]);
  58.                 LCD_data(table[h2]);
  59.                 LCD_data(table[h3]);
  60.                 LCD_data(table[h4]);
  61.                 LCD_data(table1[2]);
  62.                
  63.                 tablex((tempv*3/10)*((tempa*3-30)/10)/10);//计算功率
  64.                 zhiling(0x80+0x40);
  65.                 LCD_data(table[h1]);
  66.                 LCD_data(table[h2]);
  67.                 LCD_data(table1[0]);
  68.                 LCD_data(table[h3]);
  69.                 LCD_data(table[h4]);
  70.                 LCD_data(table1[3]);
  71.                
  72.                 tablex(maz/1000);//显示MAH
  73.                 zhiling(0x88+0x40);
  74.                 LCD_data(table[h1]);
  75.                 LCD_data(table1[0]);
  76.                 LCD_data(table[h2]);
  77.                 LCD_data(table[h3]);
  78.                 LCD_data(table[h4]);
  79.                 LCD_data(table1[4]);
  80.                
  81.         }
  82. }
  83. //-------------------------------------------------------------
  84. void delay(uint z)///////////////延时程序
  85. {
  86.    uint x,y;
  87.    for(x=z;x>0;x--)
  88.    for(y=19;y>0;y--);
  89. }
  90. void AD_init(void)/////////////////////初始化ADC
  91. {
  92. P1ASF=0xff;//P1口全部作为模拟功能AD使用
  93. ADC_RES=0;//清零转换结果寄存器高8位
  94. ADC_RESL=0;//清零转换结果寄存器低2位
  95. ADC_CONTR=0x80;//开启AD电源
  96. delay(5);//等待1ms,让AD电源稳定
  97. }

  98. uint AD_get(uchar haha )//ADC读数
  99. {
  100.         uint rew;
  101.         ADC_CONTR=haha;//开启AD转换
  102.         _nop_(); _nop_(); _nop_(); _nop_();//要经过4个CPU时钟的延时
  103.         while(!(ADC_CONTR&0x10));//等待转换完成
  104.         ADC_CONTR&=0xe7;//关闭AD转换,ADC_FLAG位由软件清0
  105.         rew=ADC_RES*4+ADC_RESL;//组合成10位
  106.         delay(1);//等待
  107.         return rew;//返回ADC值
  108. }
  109. void zhiling(uchar zhilin)//写指令
  110. {
  111.         e=0;
  112.         rs=0;
  113.         rw=0;
  114.         P0=zhilin;
  115.         delay(20);
  116.         e=1;
  117.         delay(20);
  118.         e=0;
  119. }
  120. void LCD_data(uchar shuju)//写数据
  121. {
  122.         e=0;
  123.         rs=1;
  124.         rw=0;
  125.         P0=shuju;
  126.         delay(20);
  127.         e=1;
  128.         delay(20);
  129.         e=0;
  130. }
  131. void LCD_init(void)//初始化LCD
  132. {
  133. delay(300);
  134. zhiling(0x38);
  135. delay(100);
  136. zhiling(0x38);
  137. delay(100);
  138. zhiling(0x38);
  139. delay(100);
  140. zhiling(0x38);
  141. zhiling(0x38);
  142. zhiling(0x08);
  143. zhiling(0x01);
  144. zhiling(0x06);
  145. zhiling(0x0c);
  146. }
  147. void tablex(uint tab1)//数据分解
  148. {
  149.         h1=tab1/1000;//1023//1
  150.         h2=tab1%1000/100;//023//0
  151.         h3=tab1%100/10;//23//2
  152.         h4=tab1%10;//3
  153. }
  154. uint filter(uchar dat)//中位值滤波
  155. {
  156.    uint value_buf[N];
  157.    uint count,i,j,temp;
  158.    for(count=0;count<N;count++)
  159.    {
  160.                          AD_init();//初始化ADC
  161.       value_buf[count] = AD_get(dat);//读ADC数值
  162.       delay(1);
  163.    }
  164.    for (j=0;j<N-1;j++)
  165.    {
  166.       for (i=0;i<N-j;i++)
  167.       {
  168.          if ( value_buf[i]>value_buf[i+1] )
  169.          {
  170.             temp = value_buf[i];
  171.             value_buf[i] = value_buf[i+1];
  172.             value_buf[i+1] = temp;
  173.          }
  174.       }
  175.    }
  176.    return value_buf[(N-1)/2];
  177. }
  178. void timer_init(void)//定时器初始化
  179. {
  180.         TMOD=0x01;/////////设置工作方式1
  181.         TH0=(65536-50000)/256;///////赋值
  182.         TL0=(65536-50000)%256;
  183.         EA=1;ET0=1;//开总中断;开定时器中断
  184.         TR0=1;////////启动计数器
  185. }
  186. void timer0()interrupt 1  //定时中断
  187. {
  188.         uchar t;
  189.         TR0=0;
  190.         TH0=(65536-50000)/256;///////赋初值
  191.         TL0=(65536-50000)%256;
  192.         t++;
  193.         if(t==20)
  194.         {
  195.                 t=0;
  196.                 maz+=(mas*1000)/3600;
  197.         }
  198.         TR0=1;
  199. }
复制代码
51hei.png
全部资料51hei下载地址:
单片机数字电压表.rar (20.96 KB, 下载次数: 47)
回复

使用道具 举报

ID:1 发表于 2020-11-25 15:14 | 显示全部楼层
本帖需要重新编辑补全电路原理图,源码,详细说明与图片即可获得100+黑币(帖子下方有编辑按钮)
回复

使用道具 举报

ID:852236 发表于 2020-12-2 22:15 | 显示全部楼层
等待楼主补了原理图好好学习下
回复

使用道具 举报

ID:308393 发表于 2021-3-19 16:06 | 显示全部楼层
谢谢楼主,但是是分别用两个单片机测嘛
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

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