基于单片机的频率计的C语言源代码

控制/MCU

1822人已加入

描述

  本文是基于AT89C51单片机的频率计的C源程序。该频率计主要实现的功能有如下几个:

频率计

  1. 测试功能

  它表明数字频率计所具备的全部测试功能,一般包括测频,周期,累计脉冲数,频率比,时间间隔及自较等功能。

  2. 测量范围

  它说明不同功能的有效测量范围。如测频率时,测量范围是数字频率计处于正常工作条件下,被测信号的频率范围,一般用频率的上,下限值表示,低端大部分从10HZ开始;高端因不同的频率计而异。因此高端频率是确定低,中,高速计数器的依据。在测量周期时,测量范围常用周期的最大值,最小值表示。

  3. 输入特性

  数字频率计一般有2~3个输入通道,测试不同项目时,被测信号可经不同的通道输入仪器。输入特性是表明数字式频率计于被测信号源相连的一组特性参数,通常包括以下几个方面。

  (1)输入灵敏度。通常指仪器能正常工作的最小输入电压的有效值。常用的数字频率计的灵敏度在100mV左右。

  (2)最大输入电压。指仪器所能允许的最大输入电压值,被测信号超过该值,则仪器不能保证正常工作,甚至会损坏。

  (3)输入耦合方式。仪器设置AC和DC两种耦合方式。AC耦合时,被测信号经隔直电容输入,DC耦合时,被测信号直接进入输入电路。AC耦合时适用于测量带有直流电平的信号,DC耦合适用于低频脉冲或阶跃方波信号的测量。

  (4)输入阻抗。为了减轻信号源的负载,数字式频率计一般采用高频输入阻抗。输入阻抗由输入电阻和输入电容两部分组成。

  4. 显示及工作方式

  它表明可显示的内容,显示数字的位数,所用的显示器件以及一次测量完毕显示测量结果的持续时间。有的还说明电子计数器是“不记忆”显示方式或“记忆”显示方式。

  5. 输出

  仪器可以直接输出的标准频率信号有几种,而且可以表明输出测量数据的编码方式和输出电平等。

       C语言程序
#include
#include
#define uchar unsigned char
#define uint unsigned int
uchar temp[8]={0,0,0,0,0,0,0,0};
uchar temp1[8]={0,0,0,0,0,0,0,0};
uchar T1count,timecount,T1count1,timer,yushu,yushu1;
long fre,frx;
float zhou;
bit flag;
bit flag1;
void delay(uchar);
bit result;
sbit ird=P1^1;
sbit id=P1^0;
sbit clr=P1^2;
 
sbit en=P1^5;
sbit rw=P1^6;
sbit rs=P1^7;
 
sbit rd=P3^7;
sbit kb=P1^3;
sbit kx=P1^4;
 
 
sbit A0=P3^6;
sbit A1=P3^7;
bit start;
 
uchar code tab1[]="fre:            ";
uchar code tab2[]="frx:            ";
 
void delay(uchar z)
{
       uchar x,y;
       for(x=z;x>0;x--)
              for(y=110;y>0;y--);
}                                                     
 
panduan_bz()
{                         
    rs = 0;
    rw = 1;
    en = 1;
    result = (bit)(P2&0x80);
    en = 0;
    return(result);
}
 
void write_com(uchar com)
{
    while(panduan_bz());
       rs = 0;
       rw = 0;
       en = 0;
       P2=com;
       delay(5);
       en = 1;
       delay(5);
       en = 0;
}
 
void write_dat(uchar dat)
{
    while(panduan_bz());
       rs = 1;
       rw = 0;
       en = 0;
       P2=dat;
       delay(5);
       en = 1;
       delay(5);
       en = 0;
}
 
 
void init()
{
uchar num;
en = 0;
write_com(0x38);
write_com(0x0c);
write_com(0x06);
write_com(0x01);
write_com(0x80);
for(num=0;num<16;num++)
       {
              write_dat(tab1[num]);
              delay(5);
       }
write_com(0x80+0x40);
for(num=0;num<16;num++)
       {
              write_dat(tab2[num]);
              delay(5);
       }
}
 
void init1()
{
 ird=1;
 id=1;                 
 TMOD=0x55;                         
 TH1=0;             
 TL1=0;                                   //初值为0
 TH0=0;
 TL0=0;
 TR0=1;      
 TR1=1;             
 IE=0x8a;
 RCAP2H=(65536-47850)/256;  //重装载计数器赋初值
RCAP2L=(65536-47850)%256;
ET2=1;             //开定时器2中断
EA=1;               //开总中断
TR2=1;           
}
 
void display()
{
uchar i;
        fre=(T1count*65536+TH1*256+TL1);  //频率计算
        temp[0]=fre/10000000;
        temp[1]=fre%10000000/1000000;
        temp[2]=fre%10000000%1000000%1000000/100000;
        temp[3]=fre%10000000%1000000%1000000%100000/10000;
        temp[4]=fre%10000000%1000000%1000000%100000%10000/1000;
        temp[5]=fre%10000000%1000000%1000000%100000%10000%1000/100;
        temp[6]=fre%10000000%1000000%1000000%100000%10000%1000%100/10;
        temp[7]=fre%10000000%1000000%1000000%100000%10000%1000%100%10;
        if(fre<=999)
             {
               write_com(0x80+4);
               for(i=0;i<8;i++)
               {
               write_dat(0x30+temp[i]);      //保存要显示的数到显示缓冲区
               }  
               write_dat('H');
              write_dat('z');
               write_dat(' ');
               write_dat(' ');
              }
       else if(fre>=1000)
              {
               write_com(0x80+4);
               for(i=0;i<8;i++)
               {
               write_dat(0x30+temp[i]);      //保存要显示的数到显示缓冲区
               if(i==4)
                    {
                     write_dat('.');
                     }
               }
               write_dat('K');         
               write_dat('H');
               write_dat('z');
              }
       T1count=0;
       timecount=0;        
       TH1=0;
       TL1=0;           
       TH0=0;
    TL0=0; //定时器0重新装值,保证精确(不加的话只是最多差0.001s,0.1%)
}
 
void display1()
{    
    uchar j;
       float zhou;
       zhou=((T1count1*65536+TH0*256+TL0)*1.0549);
       frx=(long)((zhou)*256);
       temp1[0]=frx/10000000;
       temp1[1]=frx%10000000/1000000;
       temp1[2]=frx%10000000%1000000%1000000/100000;
       temp1[3]=frx%10000000%1000000%1000000%100000/10000;
       temp1[4]=frx%10000000%1000000%1000000%100000%10000/1000;
       temp1[5]=frx%10000000%1000000%1000000%100000%10000%1000/100;
       temp1[6]=frx%10000000%1000000%1000000%100000%10000%1000%100/10;
       temp1[7]=frx%10000000%1000000%1000000%100000%10000%1000%100%10;
       if(frx<=999)
             {
               write_com(0x80+0x40+4);
               for(j=0;j<8;j++)
               {
               write_dat(0x30+temp1[j]);    //保存要显示的数到显示缓冲区
               }  
               write_dat('H');
              write_dat('z');
               write_dat(' ');
               write_dat(' ');
              }
       else if(frx>=1000)
              {     // frx=frx/1000;
               write_com(0x80+0x40+4);
               for(j=0;j<8;j++)
               {
               write_dat(0x30+temp1[j]);    //保存要显示的数到显示缓冲区
               if(j==4)
                    {
                     write_dat('.');
                     }
               }
               write_dat('K');         
               write_dat('H');
               write_dat('z');
              }
}
 
 
void main(void)
 {
 init();
 init1();
 while(1)
  {
      rd=0;
      ird=1;
   if(flag==1)    //标志位为1,表示进行完了一次1S记数
              {
              flag=0;
              kb=0;
              kx=1;           
              clr=0;
              ird=0;
              id=0;
              display1();
              display();
              }
              else
              {
              kb=1;
              kx=0;
              }             
    
        
  }            
 }
void t1(void) interrupt 3      // 记数器中断,加1
 {
 T1count++;
 }
 
void t0(void) interrupt 1      // 记数器中断,加1
 {
 T1count1++;
 }
void Timer2() interrupt 5      //调用定时器2,自动重装载模式
{
uchar i=0;        //定义静态变量i
TF2=0;            //定时器2的中断标志要软件清0
timecount++;           //计数标志自加1
if(timecount==20)           //判断是否到1s
       {
       timecount=0;              //将静态变量清0                                               
       flag=1;
       }
}
打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分