|
下面是單片機等精度測頻率的proteus仿真原理圖(工程文件可到本帖附件中下載):
0.png (28.3 KB, 下載次數(shù): 106)
下載附件
2017-5-7 21:11 上傳
單片機等精度測頻率源程序如下:
- //#include <reg51.h>
- #include <STC12C5A60S2.h>
- #define u8 unsigned char
- #define u16 unsigned int
- #define u32 unsigned long int
- #define vtime 3000 //定時3ms,一幀8*3=24ms,頻率=40Hz
- #define jz132 //jz132 :晶振選擇132.710400MH//jz12 :12MH
- #ifdef jz12
- //正常12MH
- #define t8s_n 90
- #define psv 4
- #else
- //132.7014MH
- #define t8s_n 1000
- #define psv 900
- #endif
- //共陰數(shù)碼管段碼表
- u8 code distable[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71,0x00};
- //位選碼表
- u8 code numi[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};
- //顯示緩存
- u8 V_ram[]={0,0,3,4,5,6,7,8};
- u8 Vm[]={0,0,3,4,5,6,7,8};
- u8 wi=0,mn=0,nt=0,wbs=0; //位選循環(huán)變量
- u8 t_l,t_h,bs=0;
- u16 t_8ms=0,t_8ms0=0;
- u32 n_plus=0,t_time=0;
- u32 n_pluss,t_timee;
- main() //m1:m0 00=標(biāo)準(zhǔn); 01=推挽; 10=輸入; 11=開漏輸出
- {
- u8 i=0,k=0;
- u32 temp0=0,temp1,temp2;
- //m1:m0 00=標(biāo)準(zhǔn); 01=推挽; 10=輸入; 11=開漏輸出
- P0M1 = 0X00;
- P0M0 = 0Xff;
- P2M1 = 0X00;
- P2M0 = 0Xff; //設(shè)定P0,P2推挽輸出
- P3M1 = 0Xff;
- P3M0 = 0X00;
- TMOD = 0X50; //設(shè)定定時器0為16位計數(shù)方式
- //TH0 = (65536-vtime )/256;
- //TL0 = (65536-vtime )%256; //賦定時器0初值
- AUXR=0x80; //選擇1T指令系統(tǒng)
- ET0 = 1; //開定時器0中斷
- TR0 = 1; //啟動定時器0計數(shù)
- //TR1=1;
- PX1=1;
- PX0=1;
- PT1=1;
- IT0=1;
- IT1=1;
- ET1=1;
- n_plus=1;
- t_time=10000;
- EX0 =1;
- EA = 1; //開總中斷
- while(1)
- { //選用11.0592MH晶振,1T指令系統(tǒng),即時鐘要比正常快12倍,所以在PROTUES下晶振要選11.0592*12=132.7014MH
- //與正常的12MH比,要乘11.0592的系數(shù),即27/(625/256)
- #ifdef jz12
- //正常12MH
- temp0=n_plus;//正常12MH
- temp1=t_time;//正常12MH
- #else
- //132.7014MH
- temp0=n_plus*27;//132.7014MH
- temp1=(t_time>>8)*625;//132.7014MH
- #endif
- temp2=temp0;
- //定位小數(shù)點
- for(i=0;i<7;i++)
- { temp2=temp2*10;
- k=temp2/t_time;
- if(k>0)
- {
- bs=i;
- break;
- }
- }
- if(k==9)bs=bs-1;
- //bs=i;
- for(i=bs;i>0;i--)
- {
- temp0=temp0*10;
- }
- wbs=6-bs; //wbs是小數(shù)點位置
- //放大
- Vm[i]=0;
- for(i=1;i<8;i++)
- {
- Vm[i]=temp0*10/temp1;
- temp0=(temp0*10)%temp1;
- }
- //去掉高位0的顯示
- for(i=0;i<6;i++)
- {
- if(Vm[i]==0) //&&(Vm[i-1]==16)&&(wbs!=i))
- {
- if(wbs==i)break;
- Vm[i]=16;
- }
- else
- {
- if(Vm[i]!=16)break;
- }
- }
- //把計算好的值放入顯示RAM
- for(i=0;i<8;i++)
- {
- V_ram[i]=Vm[i];
- }
- }
- }
- void t0_isp() interrupt 1 using 2 //13位計時中斷
- {
- u8 dm,wx;
- if(TR1)
- {
- t_8ms++;
- if(t_8ms==t8s_n) //大約8192*10=122880us
- {
- IE1=0;
- EX1=1;
- }
- }
- dm=distable[V_ram[wi]]; //取顯示段碼
- if(wi==wbs)dm=dm|0x80;
- wx=numi[wi]; //取位選碼
- P2=0x00; //關(guān)顯示
- P0=dm; //段碼賦給P0口
- P2=wx; //點亮位選的那個數(shù)碼管
- wi++;
- if(wi==8)wi=0;
- }
-
- void int0_isp() interrupt 0 using 1 //計時開始中斷
- {
- TR1=1; //計脈沖個數(shù)開始
- TR0=0; //初始化計時
- t_l=TL0;
- t_h=TH0;
- t_8ms=0;
- TR0=1; //計時開始
- EX0=0; //關(guān)中斷
- //TF1=0;
- }
- void int1_isp() interrupt 2 using 3 //計時結(jié)束中斷
- {
- TR1=0; //停止脈沖個數(shù)計數(shù)
- TR0=0; //停止計時
- EX1=0; //關(guān)中斷
- t_timee = TH0*32+TL0;
- TR0=1; //開計時中斷,因為數(shù)碼管顯示需要
- n_pluss = ((u32)nt<<16)+TH1*256+TL1; //計算脈沖個數(shù)
- t_timee =t_timee+(u32)t_8ms*8192-psv-(t_h*32)-t_l;//計算時間
- n_plus=n_pluss;
- t_time=t_timee;
- //n_pluss=0;
- //t_timee=0;
- TH1=0; //清計數(shù)脈沖寄存器
- TL1=0;
- nt=0;
- IE0=0; //清計時開始中斷的中斷源
- EX0=1; //開計時開始中斷
- }
- void t1_isp() interrupt 3 //using 1 //計時開始中斷
- …………限于本文篇幅 余下代碼請從51黑下載附件…………
復(fù)制代碼
給大家分享一些我們老師給的一些經(jīng)典的單片機程序源碼, 一共有十多個.都有詳細(xì)的注釋,然大家快速的理解每一行代碼的意思。而且有proteus仿真原理圖。大家可以直接驗證程序的對錯.

本系列所有源碼打包下載地址(含proteus仿真工程文件和源程序):
http://www.zg4o1577.cn/bbs/dpj-82474-1.html
本例程下載:
等精度測頻.rar
(65.75 KB, 下載次數(shù): 75)
2017-5-7 21:12 上傳
點擊文件名下載附件
下載積分: 黑幣 -5
|
|