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

时间:2012-08-28

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

  1. 测试功能

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

  2. 测量范围

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

  3. 输入特性

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

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

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

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

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

  4. 显示及工作方式

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

  5. 输出

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

  C语言程序

  #include<reg52.h>

  #include <intrins.h>

  #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;

  }

  }

参考文献:

[1]. AT89C51 datasheet https://www.dzsc.com/datasheet/AT89C51_810155.html.

上一篇:基于无源光网络的测试仪硬件平台的设计与实现
下一篇:多光谱偏振成像侦察系统设计实现

免责声明: 凡注明来源本网的所有作品,均为本网合法拥有版权或有权使用的作品,欢迎转载,注明出处。非本网作品均来自互联网,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责。

相关技术资料