高精度单片机频率计的设计
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
《综合课程设计》
一.数字频率计的设计
姓名:万咬春学号2005142135
一、课程设计的目的
通过本课程设计使学生进一步巩固光纤通信、单片机原理与技术的基本概念、基本理论、分析问题的基本方法;增强学生的软件编程实现能力和解决实际问题的能力,使学生能有效地将理论和实际紧密结合,拓展学生在工程实践方面的专业知识和相关技能。
二、课程设计的内容和要求
1.课程设计内容
(硬件类)频率测量仪的设计
2.课程设计要求
频率测量仪的设计
要求学生能够熟练地用单片机中定时/计数、中断等技术,针对周期性信号的特点,采用不同的算法,编程实现对信号频率的测量,将测量的结果显示在LCD 1602
上,并运用Proteus软件绘制电路原理图,进行仿真验证。
三.实验原理
可用两种方法测待测信号的频率
方法一:(定时1s测信号脉冲次数)
用一个定时计数器做定时中断,定时1s,另一定时计数器仅做计数器使用,初始化完毕后同时开启两个定时计数器,直到产生1s中断,产生1s中断后立即关闭T0和T1(起保护程序和数据的作用)取出计数器寄存器内的值就是1s内待测信号的下跳沿次数即待测信号的频率。
用相关函数显示完毕后再开启T0和T1这样即可进入下一轮测量。
原理示意图如下:
实验原理分析:
1.根据该实验原理待测信号的频率不应该大于计数器的最大值65535,也就是说待测信号应小于65535Hz。
2.实验的误差应当是均与的与待测信号的频率无关。
方法二(测信号正半周期)
对于1:1占空比的方波,仅用一个定时计数器做计数器,外部中断引脚作待测信号输入口,置计数器为外部中断引脚控制(外部中断引脚为“1”切TRx=1计数器开始计数)。
单片机初始化完毕后程序等待半个正半周期(以便准确打开TRx)打开TRx,这时只要INTx (外部中断引脚)为高电平计数器即不断计数,低电平则不计数,待信号从高电平后计数器终止计数,关闭TRx保护计数器寄存器的值,该值即为待测信号一个正半周期的单片机机器周期数,即可求出待测信号的周期:待测信号周期T=2*cnt/(12/fsoc) cnt为测得待测信号的一个正半周期机器周期数;fsoc为单片机的晶振。
所以待测信号的频率f=1/T。
原理示意图如下:
实验原理分析:
1.根据该实验原理该方法只适用于1:1占空比的方波信号,要测非1:1占空比的方波信号
2.由于有执行f=1/(2*cnt/(12/fsoc))的浮点运算,而数据类型转换时未用LCD 浮点显示,故测得的频率将会被取整,如1234.893Hz理论显示为1234Hz,测
得结果会有一定程度的偏小。
也就是说测量结果与信号频率的奇偶有一定关
系。
3.由于计数器的寄存器取值在1~65535之间,用该原理时,待测信号的频率小于单片机周期的1/12时,单片机方可较标准的测得待测信号的正半周期。
故用
该原理测得信号的最高频率理论应为fsoc/12 如12MHZ的单片机为1MHz。
而最小频率为f=1/(2*65535/(12/fsoc))如12MHZ的单片机为8Hz。
四.实验内容及步骤
1. 仿真模型的构建
数字方波频率计的设计总体可分为两个模块。
一是信号频率测量,二是将测得的频率数据显示在1602液晶显示模块上。
因此可搭建单片机最小系统构建构建频率计的仿真模型。
原理图,仿真模型的总原理图如下:
2. 液晶显示部分功能与原理分析
由于此部分并非课程的主要部分,故仅作简要原理分析
A.1602硬件接口及功能接口
//硬件接口部分********************************************************** sbit LcdRs = P2^0;
sbit LcdRw = P2^1;
sbit LcdEn = P2^2;
sfr DBPort = 0x80; //P0=0x80,P1=0x90,P2=0xA0,P3=0xB0.数据端口
//向LCD写入命令或数据
************************************************************
#define LCD_COMMAND 0 // Command
#define LCD_DATA 1 // Data
#define LCD_CLEAR_SCREEN 0x01 // 清屏
#define LCD_HOMING 0x02 // 光标返回原点
//设置显示模式************************************************************ #define LCD_SHOW 0x04 //显示开
#define LCD_HIDE 0x00 //显示关
#define LCD_CURSOR 0x02 //显示光标
#define LCD_NO_CURSOR 0x00 //无光标
#define LCD_FLASH 0x01 //光标闪动
#define LCD_NO_FLASH 0x00 //光标不闪动
//设置输入模式************************************************************ #define LCD_AC_UP 0x02 //将光标返回0x00
#define LCD_AC_DOWN 0x00 // default
#define LCD_MOVE 0x01 // 画面可平移
#define LCD_NO_MOVE 0x00 //default
B .1602初始化流程和原理框图
void LCD_Initial() { LcdEn=0; LCD_Write(LCD_COMMAND,0x38); //8位数据端口,2行显示,5*7点阵 LCD_Write(LCD_COMMAND,0x38); LCD_SetDisplay(LCD_SHOW|LCD_NO_CURSOR); //开启显示, 无光标 LCD_Write(LCD_COMMAND,LCD_CLEAR_SCREEN); //清屏 LCD_SetInput(LCD_AC_UP|LCD_NO_MOVE); //AC 递增, 画面不动 }
C . 写DDRAM 地址(原理框图如上) void GotoXY(unsigned char x, bit y) { if(y==0) LCD_Write(LCD_COMMAND,0x80|x); if(y==1) LCD_Write(LCD_COMMAND,0x80|(x-0x40)); }
D .写字符串(原理框图如上)
void Print(unsigned char *str) { while(*str!='\0') { LCD_Write(LCD_DATA,*str); str++; }
初始化流程
写DDRAM 地址,DDRAM 地
址与屏幕相对应
str
写字符串
3.程序机原理框图(关于显示函数部分不列出,只分析算法函数)
方法一:
用定时计数器T0做脉冲计数器(下跳沿有效),开始与暂停由T1控制定时计数器T1做定时中断,定时1s,定时开启置T0开始计数,定时完毕,置T0为暂停,关闭T1,读取计数数据并清空计数器,将计数数据装换为有效规范的字符串显示后再开启T0和T1,进入下一轮测量。
以下是程序的核心部分:(定时1s,取计数数,并将其转换显示出来)原理框图如下
void timer1() interrupt 3 //定时50ms
{ TH1=THCLK;
TL1=TLCLK;
if(--Cnt==0) //Cnt初值为20
{
TR0=0; TR1=0;
Cnt=CntNum;
tmp=TH0*256+TL0;
TH0=TL0=0;
Dynamic_LCD_Print();
TR0=1;
TR1=1;
}
}
方法二:
用一个定时计数器做计数器,外部中断引脚作待测信号输入口,置计数器为外部中断引脚控制(外部中断引脚为“1”切TRx=1计数器开始计数)。
单片机初始化完毕后程序等待半个正半周期(以便准确打开TRx)打开TRx,这时只要INTx(外部中断引脚)为高电平计数器即不断计数,低电平则不计数,待信号从高电平后计数器终止计数,关闭TRx保护计数器寄存器的值,该值即为待测信号一个正半周期的单片机机器周期数,即可求出待测信号的周期:待测信号周期T=2*cnt/(12/fsoc) cnt为测得待测信号的一个正半周期机器周期数;fsoc为单片机的晶振。
所以待测信号的频率f=1/T。
以下是程序的核心部分:(原理框图如下)
void chkfreq() //
{ while(FreqIN==0);
while(FreqIN==1);
TR0=1;
while(FreqIN==0);
while(FreqIN==1);
TR0=0;
cnttime=500000/(TH0*256+TL0);
TH0=TL0=0;
tmp=(int)cnttime;
Dynamic_LCD_Print();
}
4.原理框图如下
五.课程设计结果及结论
1.通过程序调试,用Protues 用两种方法均可测得小于6Mhz 的频率,以
下是用方法一测量1000Hz 频率的仿真图:
方法一流程图 方法二流程图
2.实验结果及误差分析
对于用原理一
A.待测信号的频率小于65535Hz。
B.实验的误差2000Hz时为0.05%;10000Hz时为0.07% ;50000~60000Hz 时为0.073%。
对于用原理二
C.在频率8-10000Hz时测得的值相当精确,频率为奇数时有1-2的误差。
D.超过频率8-10000Hz测得值完全错误
由此可见实验结果符合之前的原理分析,验证成功。
3。
实验优化及改进建议
a)方法一可将计数器0更改为中断扩展数据位数并延长定时时间,数据处理
后和测量大于65535Hz的频率,但由T0中断不确定性,加大了测量范围
会加大测量误差
b)方法二可将硬件待测信号取反接入剩余的外部中断接口,用于测量待测信
号的负半周期,将正半周期和负半周期数相加即为待测信号的周期。
这样
即可测量非均衡占空比的方波信号。
c)方法二还可计多次正半周期取平均值,可大大提高精度,但这样会提高实
验的最低量程
4.两算法的对比
a)方法一误差均衡,切易于扩大量程,且可测量任意占空比的方波信号,但
由于单片机的限制频率越高误差将表现更明显。
b)方法二在量程内误差比方法一稍小,占用CPU资源较小,但量程比方法
一小,切不能测量非均衡占空比的频率信号,超过量程测量结果完全错误。
c)由此可见方法一较方法二有明显的优势
六.课程设计的心得体会
通过这次综合实验,不仅加深了我对单片接的认识而却还学会了设计,开发以及实际测试,锻炼了我们的实际动手能力。
在课程设计中通过两种原理与算法是我更清晰的认识了单片机对数据的处理,
进行程序调试。
在此期间我们遇到很多麻烦,但通过仔细分析,我一次又一次品尝到了解决问题的喜悦,最终完成了实验,在测试中我们发现了自己知识的不足,通过几天的奋斗,我们学到了很多东西,最重要的是我们学会了一种精神————永不放弃。
在以后的时间里面我们会用这种精神去学习,更上一层楼。
七、课程设计参考资料
教材:1)张毅刚编.《单片机原理及应用》[M].北京:高等教育出版社,2003.
参考书:1)Optisystem 实验指导书(自编讲义)。
2)周景润编著.基于Proteus的单片机设计与仿真[M].北京:北京
航空航天出版社,2007.
附录(完整的源程序)
一.1602_Drive.h完整的库函数
/**************************************************************************
THE 1602 CHAR LCD LIB
COPYRIGHT (c) 2008 BY wanxun
File Name: 1602_Drive.h
Author: wanxun
Created: 2008/12/1
***************************************************************************/
#ifndef LCD_CHAR_1602
#define LCD_CHAR_1602
#include <intrins.h>
//硬件接口部分**********************************************************
sbit LcdRs = P2^0;
sbit LcdRw = P2^1;
sbit LcdEn = P2^2;
sfr DBPort = 0x80; //P0=0x80,P1=0x90,P2=0xA0,P3=0xB0.数据端口
//内部等待函数**********************************************************
unsigned char LCD_Wait(void)
{
LcdRs=0;
LcdRw=1; _nop_();
LcdEn=1; _nop_();
//while(DBPort&0x80);//在用Proteus仿真时,注意用屏蔽此语句,在调用GotoXY()时,会进入死循环,
//可能在写该控制字时,该模块没有返回写入完备命令,即DBPort&0x80==0x80
//实际硬件时打开此语句
LcdEn=0;
return DBPort;
}
//向LCD写入命令或数据
************************************************************
#define LCD_COMMAND 0 // Command
#define LCD_DATA 1 // Data
#define LCD_CLEAR_SCREEN 0x01 // 清屏
#define LCD_HOMING 0x02 // 光标返回原点
void LCD_Write(bit style, unsigned char input)
{
LcdEn=0;
LcdRs=style;
LcdRw=0; _nop_();
DBPort=input; _nop_();//注意顺序
LcdEn=1; _nop_();//注意顺序
LcdEn=0; _nop_();
LCD_Wait();
}
//设置显示模式************************************************************
#define LCD_SHOW 0x04 //显示开
#define LCD_HIDE 0x00 //显示关
#define LCD_CURSOR 0x02 //显示光标
#define LCD_NO_CURSOR 0x00 //无光标
#define LCD_FLASH 0x01 //光标闪动
#define LCD_NO_FLASH 0x00 //光标不闪动
void LCD_SetDisplay(unsigned char DisplayMode)
{
LCD_Write(LCD_COMMAND, 0x08|DisplayMode);
}
//设置输入模式************************************************************ #define LCD_AC_UP 0x02
#define LCD_AC_DOWN 0x00 // default
#define LCD_MOVE 0x01 // 画面可平移
#define LCD_NO_MOVE 0x00 //default
void LCD_SetInput(unsigned char InputMode)
{
LCD_Write(LCD_COMMAND, 0x04|InputMode);
}
//移动光标或屏幕************************************************************ /*
#define LCD_CURSOR 0x02
#define LCD_SCREEN 0x08
#define LCD_LEFT 0x00
#define LCD_RIGHT 0x04
void LCD_Move(unsigned char object, unsigned char direction)
{
if(object==LCD_CURSOR)
LCD_Write(LCD_COMMAND,0x10|direction);
if(object==LCD_SCREEN)
LCD_Write(LCD_COMMAND,0x18|direction);
}
*/
//初始化LCD************************************************************
void LCD_Initial()
{
LcdEn=0;
LCD_Write(LCD_COMMAND,0x38); //8位数据端口,2行显示,5*7点阵LCD_Write(LCD_COMMAND,0x38);
LCD_SetDisplay(LCD_SHOW|LCD_NO_CURSOR); //开启显示, 无光标
LCD_Write(LCD_COMMAND,LCD_CLEAR_SCREEN); //清屏
LCD_SetInput(LCD_AC_UP|LCD_NO_MOVE); //AC递增, 画面不动
}
//************************************************************************ void GotoXY(unsigned char x, bit y)
{
if(y==0)
LCD_Write(LCD_COMMAND,0x80|x);
if(y==1)
LCD_Write(LCD_COMMAND,0x80|(x-0x40));
}
void Print(unsigned char *str)
{
while(*str!='\0')
{
LCD_Write(LCD_DATA,*str);
str++;
}
}
/*
void LCD_LoadChar(unsigned char user[8], unsigned char place)
{
unsigned char i;
LCD_Write(LCD_COMMAND,0x40|(place*8));
for(i=0; i<8; i++)
LCD_Write(LCD_DATA,user[i]);
}
*/
//************************************************************************
#endif
二.方法一的源程序
#include <REG51.h>
#include <1602_Drive.h>
/******************************************************************
* 定义接口: * * 液晶显示器的接口“1602_Drive.h”库函数中已经定义 *
* 定义待测方波频率的接口: *
* P3^5(T0口)做时钟输入接口; *
******************************************************************/
//==================================================================
//用测量脉冲次数的方法时定义的定时1s的参数
#define THCLK 0x3c
#define TLCLK 0xb0
#define CntNum 20
//==================================================================
//定义中间变量
unsigned int Cnt;
unsigned int tmp;
unsigned char outcnt[8];
//==================================================================
//将测量的整数装换为标准有效的字符串
void NumToChar(void)
{ unsigned char i;
outcnt[0]=tmp/10000+48;tmp%=10000;
outcnt[1]=tmp/1000+48;tmp%=1000;
outcnt[2]=tmp/100+48;tmp%=100;
outcnt[3]=tmp/10+48;tmp%=10;
outcnt[4]=tmp+48;
outcnt[5]='H';
outcnt[6]='z';
outcnt[7]='\0';
for(i=0;i<4;i++) //将字符中数字的最高有效位之前的'0'清空为‘’。
{ if(outcnt[i]=='0')outcnt[i]=' ';
else break;
}
}
//==================================================================
//静态显示文本
void Static_LCD_Print()
{ GotoXY(0,0);
Print("Freq is:");
GotoXY(1,1);
Print("Made by wanxun");
}
//==================================================================
//动态显示数据
void Dynamic_LCD_Print()
{ NumToChar();
GotoXY(9,0);
Print(outcnt);
}
//==================================================================
/******************************************************************
*用定时计数器T0做脉冲计数器(下跳沿有效),开始与暂停由T1控制 *
*定时计数器T1做定时中断,定时1s,定时开启置T0开始计数,定时完毕置 *
*T0为暂停,关闭T1,读取计数数据并清空计数器,将计数数据装换为有效 *
*规范的字符串显示后再开启T0和T1,进入下一轮测量 *
******************************************************************/
void Initial_C51()
{ TH0=TL0=0;
TH1=THCLK;
TL1=TLCLK;
TR0=0;
TMOD=0x15;
IE=0x88;
TR1=0;
Cnt=CntNum;
}
void timer1() interrupt 3 //定时50ms
{
TH1=THCLK;
TL1=TLCLK;
if(--Cnt==0)
{
TR0=0;
TR1=0;
Cnt=CntNum;
tmp=TH0*256+TL0;
TH0=TL0=0;
Dynamic_LCD_Print();
TR0=1;
TR1=1;
}
}
void main(void)
{
Initial_C51();
LCD_Initial();
Static_LCD_Print();
TR0=1;
TR1=1;
do{ //空循环用于执行其他任务
}while(1);
}
三.方法二源程序
#include <REG51.h>
#include <1602_Drive.h>
/******************************************************************
* 定义接口: *
* 液晶显示器的接口“1602_Drive.h”库函数中已经定义 *
* 定义待测方波频率的接口: *
* *
******************************************************************/
sbit FreqIN=P3^2;
//==================================================================
//定义中间变量
unsigned int tmp;
float cnttime;
unsigned char outcnt[8];
//==================================================================
//将测量的整数装换为标准有效的字符串
void NumToChar(void)
{ unsigned char i;
outcnt[0]=tmp/10000+48;tmp%=10000;
outcnt[1]=tmp/1000+48;tmp%=1000;
outcnt[2]=tmp/100+48;tmp%=100;
outcnt[3]=tmp/10+48;tmp%=10;
outcnt[4]=tmp+48;
outcnt[5]='H';
outcnt[6]='z';
outcnt[7]='\0';
for(i=0;i<4;i++) //将字符中数字的最高有效位之前的'0'清空为‘’。
{ if(outcnt[i]=='0')outcnt[i]=' ';
else break;
}
}
//==================================================================
//静态显示文本
void Static_LCD_Print()
{ GotoXY(0,0);
Print("Loading....");
GotoXY(1,1);
Print("Made by wanxun");
}
//==================================================================
//动态显示数据
void Dynamic_LCD_Print()
{ NumToChar();
GotoXY(0,0);
Print("Freq is: ");
Print(outcnt);
}
/****************************************************************** *以下为用测量脉冲周期来测量频率的方法 * ******************************************************************/ void Initial_C51()
{ TH0=TL0=0;
TR0=0;
TMOD=0x09;
}
void chkfreq() //
{ while(FreqIN==0);
while(FreqIN==1);
TR0=1;
while(FreqIN==0);
while(FreqIN==1);
TR0=0;
cnttime=500000/(TH0*256+TL0);
TH0=TL0=0;
tmp=(int)cnttime;
Dynamic_LCD_Print();
}
void main(void)
{
Initial_C51();
LCD_Initial();
Static_LCD_Print();
chkfreq();
while(1);
}。