找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 5479|回复: 8
收起左侧

单片机高精度频率计设计 原理图+pcb+源程序

  [复制链接]
ID:60479 发表于 2016-6-29 11:01 | 显示全部楼层 |阅读模式
0.png
频率计pcb

0.png
高精度频率计原理图


单片机程序如下:
  1. /**********等精度频率计*************/
  2. /******LCD1602+STC12c5410AD+40M*****/
  3. /************2013-8-1***************/
  4. /***********0.1HZ-40MHZ*************/
  5. #include"STC12C5410AD.H"
  6. #include "stdio.h"
  7. #include"intrins.h"
  8. #define uchar unsigned char
  9. #define uint unsigned int
  10. /************端口定义***************/
  11. sbit fp = P2^2;
  12. sbit LED = P2^6;
  13. sbit lcdrs = P2^1;
  14. sbit lcden = P2^0;         
  15. sbit GATE  = P3^7;         //门控信号
  16. /************变量声明***************/
  17. uint t0,t1;
  18. double feq;        //测得频率值
  19. uint x=1000;//初始化闸门时间                                         
  20. /************1ms延时***************/
  21. void delay(uint z)
  22. {
  23.         uint x,y;
  24.         for(x=z;x>0;x--)
  25.                 for(y=1400;y>0;y--);
  26. }
  27. /**********液晶写指令*************/
  28. void write_com(uchar com)
  29. {
  30.         lcdrs=0;
  31.         lcden=0;
  32.         P1=com;
  33.         delay(1);
  34.         lcden=1;
  35.         delay(1);
  36.         lcden=0;       
  37. }
  38. /**********液晶写数据*************/
  39. void write_date(uchar date)
  40. {
  41.         lcdrs=1;
  42.         lcden=0;
  43.         P1=date;
  44.         delay(1);
  45.         lcden=1;
  46.         delay(1);
  47.         lcden=0;       
  48. }
  49. /**********液晶初始化*************/
  50. void init()
  51. {
  52.         uchar num;
  53.         uchar code table[]="f: ---Ready---- ";//初始化显示
  54.         uchar code table1[]="t: ---Ready---- ";
  55.         lcden=0;
  56.         GATE=0;                                                                //开始先关闸门保证第一次测量准确
  57.         write_com(0x38);
  58.         write_com(0x0c);
  59.         write_com(0x06);
  60.         write_com(0x01);
  61.         write_com(0x80);
  62.         for(num=0;num<15;num++)
  63.                 write_date(table[num]);
  64.         write_com(0x80+0x40);
  65.         for(num=0;num<15;num++)
  66.                 write_date(table1[num]);
  67.     TMOD=0xD9;        //T0内计数,T1外计数
  68.         AUXR=(AUXR|0x80);
  69.         AUXR=(AUXR|0x40);
  70.         AUXR=(AUXR|0x04);   
  71.         TR0=1;                               
  72.         TR1=1;
  73.         ET0=1;                                  
  74.         ET1=1;
  75.         EA=1;                //EA最后保证一起计数

  76. }
  77. /**********拆分显示*************/
  78. void write(double f)                               
  79. {
  80.         uchar i;
  81.         uchar ch[12];
  82.         sprintf(ch, "%.6f", f);       //把数转换为字符串
  83.         for(i=0;ch[i]!='\0';i++)
  84.         {
  85.            write_date(ch[i]);
  86.         }          
  87. }
  88. /*********频率计算*************/
  89. void calcu_Fx()
  90. {       
  91.         uchar i;
  92.         float N,M;
  93.         double feq_cl;
  94.         M=(t0*65536)+(TH0*256)+TL0; //内计数值
  95.         N=(t1*65536)+(TH1*256)+TL1; //外计数值
  96.         if(fp==0)                                        //判断是否分频
  97.                 feq=(N/M)*80000000;          
  98.         else
  99.                 feq=(N/M)*40000000;

  100.         feq_cl=feq;

  101.         write_com(0x80+0x03);                //第一行刷屏
  102.         for(i=0;i<16;i++)
  103.                 write_date(' ');
  104.         write_com(0x80+0x43);                //第二行刷屏
  105.         for(i=0;i<16;i++)
  106.                 write_date(' ');
  107.         if(feq>0.05&feq<=1000)                //频率显示HZ/周期S
  108.         {
  109.                 write_com(0x80+0x03);
  110.                 write(feq_cl);
  111.                 write_date('H');                                 
  112.                    write_date('z');
  113.                 write_com(0x80+0x43);
  114.                 write(1/feq_cl);                                 
  115.             write_date('s');
  116.         }
  117.         else if(feq>1000&feq<=1000000)//频率显示KHZ/周期mS
  118.         {
  119.                 write_com(0x80+0x03);
  120.                 write(feq_cl/1000);
  121.                 write_date('K');
  122.                 write_date('H');                                 
  123.                    write_date('z');
  124.                 write_com(0x80+0x43);
  125.                 write(1000/feq_cl);
  126.                 write_date('m');                                 
  127.             write_date('s');
  128.         }
  129.         else if(feq>=1000000)          //频率显示MHZ/周期uS
  130.         {
  131.                 write_com(0x80+0x03);
  132.                 write(feq_cl/1000000);
  133.                 write_date('M');
  134.                 write_date('H');                               
  135.             write_date('z');
  136.                    write_com(0x80+0x43);
  137.                 write(1000000/feq_cl);
  138.                 write_date('u');                                 
  139.             write_date('s');
  140.         }
  141.         else                                         //无输入频率显示0HZ周期显示0s
  142.         {
  143.                  write_com(0x80+0x03);
  144.                 write(0);
  145.                 write_date('H');                               
  146.             write_date('z');
  147.                    write_com(0x80+0x43);
  148.                 write(0);                                 
  149.             write_date('s');
  150.         }
  151.                
  152.                
  153. }
  154. /********自动闸门选择*************/
  155. void chane_time()
  156. {
  157.         if(feq>0.05&feq<=0.5)        //频率小于0.5HZ 闸门时间20秒                                 
  158.                 x=20000;
  159.         if(feq>0.5&feq<=10)//频率小于10HZ大于0.5HZ,闸门时间6秒
  160.                 x=8000;                                          
  161.         if(feq>10&feq<=100)//频率小于100HZ大于10HZ,闸门时间4秒
  162.                 x=6000;       
  163.         if(feq>100&feq<=10000)//频率小于2000HZ大于100HZ,闸门时间2秒
  164.                 x=4000;                                 
  165.         if(feq>10000)                  //频率大于10K闸门时间1S               
  166.                 x=2000;

  167. }

  168. /**********主程序*************/
  169. void main()
  170. {
  171.         init();                                            //测频初始化
  172.         while(1)
  173.         {
  174.                 LED=0;
  175.                 GATE=1;                                         //开闸门
  176.                 delay(2*x);                          //延时关门时间(第一次1秒)
  177.                 GATE=0;                                 //闸门时间到gate为0;关门然后计算
  178.                 calcu_Fx();                         //计算频率
  179.                 TH1=TL1=TH0=TL0=t1=t0=0;  //所以计数清零为下次做准备       
  180.                 chane_time();            //根据频率选择闸门时间
  181.         }
  182. }
  183. /*******定时器0对内计数*********/
  184. void timer0() interrupt 1
  185. {
  186.         t0++; //内部计数
  187. }
  188. /*******定时器1对外计数*********/
  189. void timer1() interrupt 3
  190. {
  191.         t1++;//外部计数       
  192. }
复制代码


0.png

高精度频率.rar

482.98 KB, 下载次数: 152, 下载积分: 黑币 -5

评分

参与人数 2黑币 +62 收起 理由
YJGG + 12 共享资料的黑币奖励!
admin + 50 共享资料的黑币奖励!

查看全部评分

回复

使用道具 举报

ID:150693 发表于 2017-4-6 11:56 | 显示全部楼层
谢谢楼主的分享
回复

使用道具 举报

ID:194701 发表于 2017-5-1 09:02 | 显示全部楼层
谢谢楼主的分享
回复

使用道具 举报

ID:192020 发表于 2017-5-14 10:47 来自手机 | 显示全部楼层
感谢楼主分享
回复

使用道具 举报

ID:196311 发表于 2017-5-15 11:03 | 显示全部楼层
谢谢楼主,下载学习学习
回复

使用道具 举报

ID:308371 发表于 2018-4-15 08:10 | 显示全部楼层
feq=(N/M)*80000000;  这句是什么意思啊???
回复

使用道具 举报

ID:687653 发表于 2020-5-16 20:57 | 显示全部楼层
您好,请问这频率计最高测量范围可达多少
回复

使用道具 举报

ID:102702 发表于 2022-4-16 00:40 | 显示全部楼层
楼主能说明一下具体功能和测量参数吗?
回复

使用道具 举报

ID:1024217 发表于 2022-5-6 08:07 来自手机 | 显示全部楼层
怎样测LC振荡频率?
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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