找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 6827|回复: 2
打印 上一主题 下一主题
收起左侧

单片机等精度测频率源程序 原理图与Proteus仿真 代码注释很详细

[复制链接]
跳转到指定楼层
楼主

下面是单片机等精度测频率的proteus仿真原理图(工程文件可到本帖附件中下载):


单片机等精度测频率源程序如下:
  1. //#include <reg51.h>
  2. #include <STC12C5A60S2.h>
  3. #define u8 unsigned char
  4. #define u16 unsigned   int
  5. #define u32 unsigned long  int
  6. #define vtime        3000   //定时3ms,一帧8*3=24ms,频率=40Hz
  7. #define  jz132                                //jz132 :晶振选择132.710400MH//jz12 :12MH
  8. #ifdef        jz12
  9.                 //正常12MH
  10.         #define         t8s_n        90
  11.         #define         psv                4
  12. #else
  13.                 //132.7014MH
  14.         #define         t8s_n        1000
  15.         #define         psv                900
  16. #endif


  17. //共阴数码管段码表
  18. u8 code distable[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71,0x00};
  19. //位选码表
  20. u8  code numi[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};
  21. //显示缓存
  22. u8 V_ram[]={0,0,3,4,5,6,7,8};
  23. u8 Vm[]={0,0,3,4,5,6,7,8};

  24. u8        wi=0,mn=0,nt=0,wbs=0;  //位选循环变量
  25. u8                t_l,t_h,bs=0;
  26. u16                t_8ms=0,t_8ms0=0;
  27. u32                n_plus=0,t_time=0;
  28. u32                n_pluss,t_timee;
  29. main()                                                 //m1:m0  00=标准;   01=推挽;   10=输入;   11=开漏输出
  30. {
  31.         u8        i=0,k=0;
  32.         u32                temp0=0,temp1,temp2;
  33.         //m1:m0  00=标准;   01=推挽;   10=输入;   11=开漏输出
  34.         P0M1 = 0X00;
  35.         P0M0 = 0Xff;
  36.         P2M1 = 0X00;
  37.         P2M0 = 0Xff; //设定P0,P2推挽输出
  38.         P3M1 = 0Xff;
  39.         P3M0 = 0X00;

  40.           TMOD = 0X50; //设定定时器0为16位计数方式

  41.           //TH0 = (65536-vtime )/256;
  42.           //TL0 = (65536-vtime )%256; //赋定时器0初值
  43.         AUXR=0x80; //选择1T指令系统
  44.           ET0 = 1;  //开定时器0中断
  45.            TR0 = 1;  //启动定时器0计数
  46.         //TR1=1;
  47.         PX1=1;
  48.         PX0=1;
  49.         PT1=1;
  50.         IT0=1;
  51.         IT1=1;
  52.         ET1=1;
  53.                 n_plus=1;
  54.                 t_time=10000;

  55.         EX0 =1;
  56.           EA = 1;          //开总中断
  57.         while(1)
  58.         {                   //选用11.0592MH晶振,1T指令系统,即时钟要比正常快12倍,所以在PROTUES下晶振要选11.0592*12=132.7014MH
  59.                                 //与正常的12MH比,要乘11.0592的系数,即27/(625/256)
  60.                 #ifdef        jz12
  61.                 //正常12MH
  62.                         temp0=n_plus;//正常12MH
  63.                         temp1=t_time;//正常12MH
  64.                 #else
  65.                 //132.7014MH
  66.                         temp0=n_plus*27;//132.7014MH
  67.                         temp1=(t_time>>8)*625;//132.7014MH
  68.                 #endif
  69.                 temp2=temp0;
  70.                 //定位小数点
  71.                 for(i=0;i<7;i++)
  72.                 {        temp2=temp2*10;
  73.                         k=temp2/t_time;
  74.                         if(k>0)
  75.                         {
  76.                                 bs=i;
  77.                                 break;
  78.                         }

  79.                 }
  80.                 if(k==9)bs=bs-1;
  81.                 //bs=i;
  82.                 for(i=bs;i>0;i--)
  83.                 {
  84.                         temp0=temp0*10;
  85.                 }
  86.                 wbs=6-bs; //wbs是小数点位置
  87.                 //放大
  88.                 Vm[i]=0;
  89.                 for(i=1;i<8;i++)
  90.                 {
  91.                          Vm[i]=temp0*10/temp1;
  92.                         temp0=(temp0*10)%temp1;
  93.                 }
  94.                 //去掉高位0的显示
  95.                 for(i=0;i<6;i++)
  96.                 {
  97.                         if(Vm[i]==0)        //&&(Vm[i-1]==16)&&(wbs!=i))
  98.                         {
  99.                                 if(wbs==i)break;
  100.                                 Vm[i]=16;
  101.                         }
  102.                         else
  103.                         {
  104.                                 if(Vm[i]!=16)break;
  105.                         }
  106.                 }
  107.                 //把计算好的值放入显示RAM
  108.                 for(i=0;i<8;i++)
  109.                 {
  110.                                 V_ram[i]=Vm[i];
  111.                 }


  112.         }
  113. }

  114. void t0_isp() interrupt 1         using                2         //13位计时中断
  115. {
  116.         u8 dm,wx;
  117.         if(TR1)
  118.         {
  119.                 t_8ms++;
  120.                 if(t_8ms==t8s_n)        //大约8192*10=122880us
  121.                 {
  122.                         IE1=0;
  123.                         EX1=1;
  124.                 }
  125.         }
  126.           dm=distable[V_ram[wi]];        //取显示段码
  127.         if(wi==wbs)dm=dm|0x80;
  128.         wx=numi[wi];                           //取位选码
  129.         P2=0x00;                           //关显示
  130.         P0=dm;                                   //段码赋给P0口
  131.         P2=wx;                                   //点亮位选的那个数码管
  132.         wi++;
  133.         if(wi==8)wi=0;
  134. }
  135.   
  136. void int0_isp() interrupt 0           using        1           //计时开始中断
  137. {
  138.                 TR1=1;          //计脉冲个数开始
  139.                 TR0=0;          //初始化计时
  140.                 t_l=TL0;
  141.                 t_h=TH0;
  142.                 t_8ms=0;
  143.                 TR0=1;         //计时开始
  144.                 EX0=0;         //关中断
  145.                 //TF1=0;
  146. }
  147. void int1_isp() interrupt 2                using                3                //计时结束中断
  148. {
  149.                 TR1=0;          //停止脉冲个数计数
  150.                 TR0=0;          //停止计时
  151.                 EX1=0;          //关中断
  152.                 t_timee = TH0*32+TL0;
  153.                 TR0=1;         //开计时中断,因为数码管显示需要
  154.                 n_pluss = ((u32)nt<<16)+TH1*256+TL1;         //计算脉冲个数
  155.                 t_timee =t_timee+(u32)t_8ms*8192-psv-(t_h*32)-t_l;//计算时间
  156.                 n_plus=n_pluss;
  157.                 t_time=t_timee;
  158.                 //n_pluss=0;
  159.                 //t_timee=0;
  160.                 TH1=0;         //清计数脉冲寄存器
  161.                 TL1=0;
  162.                 nt=0;
  163.                 IE0=0;         //清计时开始中断的中断源
  164.                 EX0=1;         //开计时开始中断
  165. }

  166. void t1_isp() interrupt 3           //using        1           //计时开始中断



  167. …………限于本文篇幅 余下代码请从51黑下载附件…………
复制代码

给大家分享一些我们老师给的一些经典的单片机程序源码, 一共有十多个.都有详细的注释,然大家快速的理解每一行代码的意思。而且有proteus仿真原理图。大家可以直接验证程序的对错.


本系列所有源码打包下载地址(含proteus仿真工程文件和源程序):
http://www.51hei.com/bbs/dpj-82474-1.html

本例程下载:
等精度测频.rar (65.75 KB, 下载次数: 75)

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏6 分享淘帖 顶 踩
回复

使用道具 举报

沙发
ID:639106 发表于 2020-7-18 20:10 | 只看该作者
很好很好,感谢楼主的好资料
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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