介绍计数器 定时器 程序 的编写步骤 (C语言的)

介绍计数器 定时器 程序 的编写步骤 (C语言的),第1张

假设你用的晶振为12m,用p10口输出周期为2ms的方波。使用定时器工作方式1

至于计数初值的计算,授之以鱼不如授之以渔!

在定时器模式下,计数器的计数脉冲来自于晶振脉冲的12分频信号,即对机器周期进行计数。若选择12m晶振,则定时器的计数频率为1mhz。假设定时时间为t,机器周期为t1,即12/晶振频率。x为定时器初值。则

x=2^n-t/t1。方式0,n=13,方式1时,n=16,方式2和方式3,n=8

自己算去吧!

#include

void

inittimer0(void)//

{

tmod

=

0x01;

th0

=

0x0fc;

//计数器初值

tl0

=

0x18;

ea

=

1;

et0

=

1;

tr0

=

1;//开启定时器t0

}

void

main(void)

{

inittimer0();

}

void

timer0interrupt(void)

interrupt

1

{

th0

=

0x0fc;//重新赋初值

tl0

=

0x18;

p10=~p10;

//输出方波

}

是51单片机吗? 如果是51单片机,你可以使用TMOD设置两个定时器,然后使用TR0、TR1分别启动两个定时器。PT0置为“1”,就可以中断定时器1的服务函数,相反,如果PT1置为“1”就可以中断定时器0的服务函数。 如果不设置PT的值,默认为“0”,这样的话,T0和T1谁也不能中断谁。

在精度要求较高的情况下,如要求误差不大于1ms时,可以利用GetTickCount()函数。该函数的返回值是DWORD型,表示以ms为单位的计算机启动后经历的时间间隔。下列的代码可以实现50ms的精确定时,其误差小于1ms。

// 起始值和中止值

DWORD dwStart, dwStop ;

dwStop = GetTickCount();

while(TRUE) {

// 上一次的中止值变成新的起始值

dwStart = dwStop ;

// 此处添加相应控制语句

do

{

dwStop = GetTickCount() ;

}while(dwStop - 50 < dwStart) ;

}

微软公司在其多媒体Windows中提供了精确定时器的底层API支持。利用多媒体定时器可以很精确地读出系统的当前时间,并且能在非常精确的时间间隔内完成一个事件、函数或过程的调用。利用多媒体定时器的基本功能,可以通过两种方法实现精确定时。

1使用timeGetTime()函数

该函数定时精度为ms级,返回从Windows启动开始所经过的时间。由于使用该函数是通过查询的方式进行定时控制的,所以,应该建立定时循环来进行定时事件的控制。

2 使用timeSetEvent()函数

利用该函数可以实现周期性的函数调用。函数的参数说明如下:

uDelay:延迟时间;

uResolution:时间精度,在Windows中缺省值为1ms;

lpFunction:回调函数,为用户自定义函数,定时调用;

dwUser:用户参数;

uFlags:标志参数;

TIME_ONESHOT:执行一次;

TIME_PERIODIC:周期性执行。

具体应用时,可以通过调用timeSetEvent()函数,将需要周期性执行的任务定义在lpFunction回调函数中(如:定时采样、控制等),从而完成所需处理的事件。需要注意的是:任务处理的时间不能大于周期间隔时间。另外,在定时器使用完毕后,应及时调用timeKillEvent()将之释放

对于精确度要求更高的定时 *** 作,则应该使用QueryPerformanceFrequency()和QueryPerformanceCounter()函数。这两个函数是系统提供的精确时间函数,并要求计算机从硬件上支持精确定时器。QueryPerformanceFrequency()函数和QueryPerformanceCounter()函数的原型如下:

BOOL QueryPerformanceFrequency(LARGE_INTEGER *lpFrequency);

BOOL QueryPerformanceCounter(LARGE_INTEGER *lpCount);

数据类型LARGE_INTEGER既可以是一个8字节长的整型数,也可以是两个4字节长的整型数的联合结构,其具体用法根据编译器是否支持64位而定。

在进行定时之前,先调用QueryPerformanceFrequency()函数获得机器内部定时器的时钟频率,然后在需要严格定时的事件发生之前和发生之后分别调用QueryPerformanceCounter()函数,利用两次获得的计数之差及时钟频率,计算出事件经历的精确时间。

(这是我同学编的,拿来给你)

#include <reg51h>

#include <intrinsh>

unsigned char i;

void main()

{

TMOD=0x00;

TH1=0x3c;

TL1=0xb0;

EA=1;

ET1=1;

TR1=1;

while(1)

{

if(i==20)

{

i=0;

//下面写你要处理的的其他程序,你只要了定时1s。

}

}

}

void a(void) interrupt 3

{

unsigned char i;

TH1=0x3c;

TL1=0xb0;

i++;

}

#include<reg51h>

#include<intrinsh>

#define uchar unsigned char

uchar keyval;

uchar led1=0xfe,led2=0x55,led3=0x0f;

sbit key=P1^0;

void t0isr() interrupt 1

{

TH0=(65536-60000)/256;

TL0=(65536-60000)%256;

switch(keyval)

{

case 1:

P0=led1;

led1=_crol_(led1,1)

break;

case 2:

P0=led2;

led2=~led2;

break;

case 3:

P0=led3;

led3=~led3;

break;

default:break;

}

}

main()

{

TMOD=0x01;

TH0=(65536-60000)/256;

TL0=(65536-60000)%256;

TR0=1;

ET0=1;

EA=1;

while(1)

{

if(key==0)

{

while(key==0);

keyval++;

keyval%=4;

}

}

}

#include<reg51h>

#define uchar unsigned char

uchar pwm=50,cnt;

sbit pluse=P1^0;

sbit keyu=P1^4;

sbit keyd=P1^5;

void t0isr() interrupt 1

{

TH0=(65536-100)/256;

TL0=(65536-100)%256;

cnt++;

if(pwm>0)

{

if(cnt>100)cnt=0;

if(cnt<=pwm)pluse=1;

else pluse=0;

}

else pluse=0;

}

main()

{

TMOD=0x01;

TH0=(65536-100)/256;

TL0=(65536-100)%256;

TR0=1;

ET0=1;

EA=1;

while(1)

{

if(keyu==0)

{

while(keyu==0);

pwm++;

if(pwm>100)pwm=100;

}

if(keyd==0)

{

while(keyd==0);

if(pwm>0)pwm--;

}

}

}

 //一个示例程序。

#include<stdlibh>

#include<unistdh>

#include<signalh>

#include<timeh>

#include<sys/timeh>

#define N 100 //设置最大的定时器个数

int i=0,t=1; //i代表定时器的个数;t表示时间,逐秒递增

 

struct Timer //Timer结构体,用来保存一个定时器的信息

{

    int total_time; //每隔total_time秒

    int left_time; //还剩left_time秒

    int func; //该定时器超时,要执行的代码的标志

}myTimer[N]; //定义Timer类型的数组,用来保存所有的定时器

 

void setTimer(int t,int f) //新建一个计时器

{

    struct Timer a;

    atotal_time=t;

    aleft_time=t;

    afunc=f;

    myTimer[i++]=a;

}

 

void timeout() //判断定时器是否超时,以及超时时所要执行的动作

{

    printf("Time: %d\n",t++);

    int j;

    for(j=0;j<i;j++)

    {

       if(myTimer[j]left_time!=0)

           myTimer[j]left_time--;

       else

       {

           switch(myTimer[j]func)

           {      //通过匹配myTimer[j]func,判断下一步选择哪种 *** 作

           case 1:

              printf("------Timer 1: --Hello Aillo!\n");break;

           case 2:

              printf("------Timer 2: --Hello Jackie!\n");break;

           case 3:

              printf("------Timer 3: --Hello PiPi!\n");break;

           }

           myTimer[j]left_time=myTimer[j]total_time; //循环计时

       }

    }

}

 

int main() //测试函数,定义三个定时器

{

    setTimer(3,1);

    setTimer(4,2);

    setTimer(5,3);

    signal(SIGALRM,timeout); //接到SIGALRM信号,则执行timeout函数

   

    while(1)

    {

       sleep(1); //每隔一秒发送一个SIGALRM

       kill(getpid(),SIGALRM);

    }

    exit(0);

}

以上就是关于介绍计数器 /定时器 程序 的编写步骤 (C语言的)全部的内容,包括:介绍计数器 /定时器 程序 的编写步骤 (C语言的)、求教怎么在C语言中使用定时器、如何用C语言实现精确软件定时等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!

欢迎分享,转载请注明来源:内存溢出

原文地址: https://outofmemory.cn/zz/9822328.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-02
下一篇 2023-05-02

发表评论

登录后才能评论

评论列表(0条)

保存