打印
[51单片机]

51做的频率测速程序误差控制咨询

[复制链接]
1679|11
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
mokedeng|  楼主 | 2013-9-15 22:29 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
void IntToStr(unsigned int t, unsigned char *str, unsigned char n)
{
        unsigned char a[5]; char i, j;                                         
        a[0]=(t/10000)%10;         //取得整数值到数组                 
        a[1]=(t/1000)%10;                                             
        a[2]=(t/100)%10;                                             
        a[3]=(t/10)%10;                                               
        a[4]=(t/1)%10;                                                

        for(i=0; i<5; i++)         //转成ASCII码                     
                a=a+'0';                                            
        for(i=0; a=='0' && i<=3; i++);                             
        for(j=5-n; j<i; j++)       //填充空格                        
                { *str=' ';  str++; }                                    
        for(; i<5; i++)                                               
                { *str=a; str++; }  //加入有效的数字                  
        *str='\0';
}




//count interrupt
void t0(void) interrupt 1 using 0
{
        T0count++;
        TH0=0;
        TL0=0;
}

//time interrupt
void t1(void) interrupt 3 using 0
{
        TH1=0x5d;
        TL1=0x3d;
        timecount++;
        if(timecount==40)      //1秒
              flag=1;
}





main()
{
        unsigned char i;
        init();

        while(1)
         {
             if(flag==1)    //如果定时时间到了1s
               {
                 flag=0;   //标志位清零
                 x=T0count*65536+TH0*256+TL0;  //获得整型的频率值,T0count计数器在1s内溢出的次数,每溢出一次就计数了T0count*65536次                                                                                                                                         _                                                                         //  再加上当前计数寄存器的值即为实际计数总数

                 IntToStr(x, &TempBuffer[0], 5);

                         while(TempBuffer != '\0')
                         {
                                  write_date(TempBuffer);
                                i++;
                                delay(200);   //延时200ms
                         }
                         write_com(0x80+0x48);

                 timecount=0;
                 T0count=0;
                 TH0=0;
                 TL0=0;
                 TR0=1;
                         TR1=1;
                         i = 0;
               }
           }
               
}



这个是我写的一个关于51单片机的一个计数器程序。
最高可以计数到65536,但是到1KHz或者更大的时候有误差100多,当频率越高误差越大。
请问一下各位,这个误差是出在哪些地方的,通过改进哪里可以提高误差的,希望高手能指点一下
谢谢了

相关帖子

沙发
cjseng| | 2013-9-15 23:52 | 只看该作者
注意:程序执行是需要时间的,当信号频率较高时,指令执行占用的时间就不可忽视了。

使用特权

评论回复
板凳
gonggu8181| | 2013-9-16 07:13 | 只看该作者
把取计数放到定时器中断里,用 < 替代 * ,先试试。

使用特权

评论回复
地板
ayb_ice| | 2013-9-16 07:53 | 只看该作者
测周期好点,但不是测一个周期,是很多个周期求平均值,周期计数定时器溢出计数,不能固定在<=65536

使用特权

评论回复
5
食肉狼| | 2013-9-16 12:50 | 只看该作者
蛋疼,学生写出这样的软件,很棒了。不用搞得太深

使用特权

评论回复
6
mokedeng|  楼主 | 2013-9-16 14:11 | 只看该作者
cjseng 发表于 2013-9-15 23:52
注意:程序执行是需要时间的,当信号频率较高时,指令执行占用的时间就不可忽视了。 ...

那你的意思是占用时间上要算好时间么?
能给我大概的说说么
谢谢

使用特权

评论回复
7
mokedeng|  楼主 | 2013-9-16 14:13 | 只看该作者
gonggu8181 发表于 2013-9-16 07:13
把取计数放到定时器中断里,用 < 替代 * ,先试试。

x=T0count*65536+TH0*256+TL0;   
你说的是这个计数值么
用<代替是什么意思呢  麻烦你跟我讲讲谢谢

使用特权

评论回复
8
cjseng| | 2013-9-16 15:17 | 只看该作者
你在计算x=T0count*65536+TH0*256+TL0要花费大量时间,而且,在此期间,TH0、TL0还在继续计数。

使用特权

评论回复
9
mokedeng|  楼主 | 2013-9-16 16:53 | 只看该作者
cjseng 发表于 2013-9-16 15:17
你在计算x=T0count*65536+TH0*256+TL0要花费大量时间,而且,在此期间,TH0、TL0还在继续计数。 ...

那要怎么避免

使用特权

评论回复
10
gonggu8181| | 2013-9-16 23:10 | 只看该作者
本帖最后由 gonggu8181 于 2013-9-16 23:14 编辑
#define u8  unsigned char
#define u16 unsigned int
u8 i;

void IntToStr(unsigned int t, unsigned char *str, unsigned char n)
{
        unsigned char a[5]; char i, j;                                         
        a[0]=(t/10000)%10;   
        a[1]=(t/1000)%10;                                             
        a[2]=(t/100)%10;                                             
        a[3]=(t/10)%10;                                               
        a[4]=(t/1)%10;
        for(i=0; i<5; i++)
                a=a+'0';                                            
        for(i=0; a=='0' && i<=3; i++);
        for(j=5-n; j<i; j++)               
        { *str=' ';  str++; }                                    
        for(; i<5; i++)
        { *str=a; str++; }
        *str='\0';
}
//count interrupt
void t0(void) interrupt 1
{
        T0count++;
        TH0=0;TL0=0;
}
//time interrupt
void t1(void) interrupt 3                                
{
        TH1=0x5d;TL1=0x3d;                                                //重装值
        timecount++;
        if(timecount==40)                                                      //1S
        {
                TR0=0;                                                               //停止计数
                x=T0count<<16+TH0<<8+TL0;                        //取计数值
                IntToStr(x, &TempBuffer[0], 5);                //
                while(TempBuffer != '\0')
                {
                        write_date(TempBuffer);
                        i++;
                        delay(200);
                }
                write_com(0x80+0x48);
                timecount=0;T0count=0;i = 0;
                TH0=0;TL0=0;
                TR1=1;                                                                //开始计时
                TR0=1;                                                                //开始计数
        }
}
int main(void)
{
        init();
        while(1);
}

使用特权

评论回复
11
yzgwxflove| | 2013-9-17 08:46 | 只看该作者
我只是看到貌似TH0=0;TL0=0;是多此一举,:lol

使用特权

评论回复
12
cjseng| | 2013-9-17 12:07 | 只看该作者
mokedeng 发表于 2013-9-16 16:53
那要怎么避免

你的思路就是数一秒钟内的脉冲个数,这样是可以的,不过首先你要确保这个“1秒”要精确,显然你的程序没有体现出这一点。

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

20

主题

38

帖子

0

粉丝