C语言实现电子秒表

 更新时间:2022年06月14日 11:27:22   作者:小辉_Super  
这篇文章主要为大家详细介绍了C语言实现电子秒表,毫秒级秒表,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

秒表是我们生活中常见的计时工具,特别是在运动会等比赛中,今天我就来写一个简单的电子秒表。

实现思路

这里简单介绍一下我的实现思路:

1、简单版:简单版本只实现了单次计时功能,即每次开启程序后就开始计时,如果按下键盘任意键,就结束计时,计时通过Sleep(1000)延时实现,每过1秒,计数值(总秒数)cnt加1,打印时,将总秒数cnt转换成时分秒进行显示。【Sleep()函数并不准确,只能实现粗略延时】

2、高级版:实现毫秒级计时,可重复计时(暂停、清零),计时使用gettimeofday()函数,用来获取系统的秒数和毫秒数,将计时开始和计时暂停的秒数相减,即可获得计时期间的秒数。细节请看代码部分。

简易版本

#include <stdio.h>
#include <conio.h>              //kbhit()/_kbhit()
#include <Windows.h>            //Sleep(ms)

int main()
{
    int hour = 0, min = 0, sec = 0;
    int cnt = 0;

    printf("按任意键停止计时\n");
    while(!_kbhit())            //任意键退出循环(结束计时)
    {
        hour = cnt / 3600;      //获取计时小时数
        min = cnt / 60;          //获取计时分钟数
        sec = cnt % 60;         //获取计时秒数
        printf("  %02d:%02d:%02d\r", hour, min, sec);
        Sleep(1000);            //1s延时
        cnt++;
    }
    printf("\n程序退出\n");
    return 0;
}

运行效果:

高级版本

代码可能一般,但至少功能已经实现,仅供参考

下面给出几个注释的解释:

  • 计时初始时间指的是开始计时或继续计时时的系统时间(第一次暂停后,如果继续计时,此时的系统时间即为新的计时初始时间)
  • 当前累计计时时长指的是从开始计时到当前时刻的时间差,即真正的有效计时时长
  • 总累计计时时长指的是计时初始时间之前的计时时间,这个值只有在计时暂停时才进行更新(如第一次暂停时,总累计计时时长 = 第一次暂停的系统时间 - 开始计时时的系统时间;第二次暂停时,总累计计时长 = 总累计计时时长 + 第二次在暂停的系统时间 - 上次继续计时时的系统时间…)
#include <stdio.h>
#include <conio.h>                 //kbhit()/_kbhit() getch()
#include <Windows.h>               //Sleep(ms)
#include <sys/time.h>              //struct timeval
#include <unistd.h>                //struct timeval

/******************************************************************************
 * @brief       获取系统当前秒数和毫秒(1970-1-1 0:0:0到现在)
 * @param tv    timeval结构体变量
 * @param tv_s  返回的秒数
 * @param tv_ms 返回的毫秒数
 ******************************************************************************/
void Get_Current_Timeval(struct timeval *tv, long *tv_s, long *tv_ms)
{
    gettimeofday(tv,NULL);      //获取1970-1-1到现在的时间保存到timeval变量
    *tv_s = tv->tv_sec;         //获取秒
    *tv_ms = tv->tv_usec / 1000;//获取毫秒
}

/******************************************************************************
 * @brief            获取两个timeval成员的差值,通过tv_s_diff和tv_ms_diff返回
 * @param tv_s_cur   当前系统时间秒数
 * @param tv_ms_cur  当前系统时间毫秒数
 * @param tv_s_old   计时初始时间(s)
 * @param tv_ms_old  计时初始时间(ms)
 * @param tv_s_diff  秒数的差值
 * @param tv_ms_diff 毫秒的差值
 ******************************************************************************/
void Get_Diff_Timeval(long tv_s_cur, long tv_ms_cur,\
                      long tv_s_old, long tv_ms_old,\
                      long *tv_s_diff, long *tv_ms_diff)
{
    if(tv_ms_cur < tv_ms_old)
    {
        *tv_ms_diff = tv_ms_cur + 1000 - tv_ms_old;  //获取这段时间的毫秒数
        *tv_s_diff = tv_s_cur - tv_s_old - 1; //获取这段时间的秒数(自上次暂停或自初始时间)
    }
    else
    {
        *tv_ms_diff = tv_ms_cur - tv_ms_old;  //获取这段时间的毫秒数(自上次暂停或自初始时间)
        *tv_s_diff = tv_s_cur - tv_s_old;     //获取这段时间的秒数(自上次暂停或自初始时间)
    }
}

/******************************************************************************
 * 主函数
 * ****************************************************************************/
int main(void)
{
    struct timeval tv;
    long tv_s_cur = 0, tv_ms_cur = 0;   //当前系统时间
    long tv_s_old = 0, tv_ms_old = 0;   //计时初始时间
    long tv_s_diff = 0, tv_ms_diff = 0; //存放时间的差值
    int sec_cnt = 0, msec_cnt = 0;      //当前累计计时时长
    int hour = 0, min = 0, sec = 0, msec = 0;
    int timer_step = 0;                 //计时步骤 0:未开始,
                                        //1:开始,2:暂停
    char key = 0;

    /**************** 菜单打印 ****************/
    printf("================================\n"); //菜单
    printf("| 空格:开始/暂停 R:清零 Q:退出 |\n");
    printf("================================\n");
    printf("\t%02d:%02d:%02d %02d\r", 0, 0, 0, 0);
    while(1)
    {
        /**************** 键盘按键扫描+操作 ****************/
        key = 0;
        if(_kbhit())                      //检测到按键按下
            key = getch();                //读取按键
        switch(key)
        {
            case ' ':                     //按空格键开始/暂停计时
                if(timer_step == 0)       //如果还未开启计时
                {
                    //获取当前秒和毫秒作为计时初始时间
                    Get_Current_Timeval(&tv, &tv_s_old, &tv_ms_old);
                    timer_step = 1;       //开始计时
                }
                else if(timer_step == 1)  //如果正在计时
                {
                    timer_step = 2;       //暂停计时
                    //获取当前秒和毫秒
                    Get_Current_Timeval(&tv, &tv_s_cur, &tv_ms_cur);
                    //获取当前系统时间与计时初始时间的差值
                    Get_Diff_Timeval(tv_s_cur, tv_ms_cur, tv_s_old,\
                                     tv_ms_old, &tv_s_diff, &tv_ms_diff);
                    msec_cnt += tv_ms_diff;       //更新总累计计时时长(ms)
                    if(msec_cnt >= 1000)
                    {
                        msec_cnt -= 1000;
                        sec_cnt += tv_s_diff + 1; //更新总累计计时时长(s)
                    }
                    else
                        sec_cnt += tv_s_diff;
                }
                else if(timer_step == 2)
                {
                    timer_step = 1;                //继续计时
                    //获取当前秒和毫秒
                    Get_Current_Timeval(&tv, &tv_s_cur, &tv_ms_cur);
                    tv_s_old = tv_s_cur;           //更新计时初始时间(s)
                    tv_ms_old = tv_ms_cur;         //更新计时初始时间(ms)
                }
                break;
            case 'r':                     //按r/R清零计时时间
            case 'R':
                sec_cnt = msec_cnt = 0;   //总累计计时值清零
                tv_s_old = tv_s_cur;      //更新计时初始时间(s)
                tv_ms_old = tv_ms_cur;    //更新计时初始时间(ms)
                timer_step = 0;           //回到步骤0(未开始计时)
                printf("\t%02d:%02d:%02d %02d\r", 0, 0, 0, 0);
                break;
            case 'q':
            case 'Q': printf("程序退出\n");return 0;
        }

        /**************** 计时操作 ****************/
        if(timer_step == 1)
        {
            //获取当前秒和毫秒
            Get_Current_Timeval(&tv, &tv_s_cur, &tv_ms_cur);
            //获取当前系统时间与计时初始时间的差值
            Get_Diff_Timeval(tv_s_cur, tv_ms_cur, tv_s_old,\
                             tv_ms_old, &tv_s_diff, &tv_ms_diff);
            tv_ms_diff += msec_cnt;          //当前累计计时时长(ms)
            if(tv_ms_diff >= 1000)
            {
                tv_ms_diff -= 1000;
                tv_s_diff += sec_cnt + 1;    //当前累计计时时长(s)
            }
            else
                tv_s_diff += sec_cnt;
            hour = tv_s_diff / 3600;         //获取计时小时数
            min = tv_s_diff /60;             //获取计时分钟数
            sec = tv_s_diff % 60;            //获取计时秒数
            msec = tv_ms_diff / 10;          //获取毫秒(单位10ms)
            //打印当前累计计时时长
            printf("\t%02d:%02d:%02d %02d\r", hour, min, sec, msec);

        }
        Sleep(10);       //10ms延时,防止打印太快导致显示效果不佳
    }
    return 0;
}

运行效果:

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • C语言数据结构与算法之图的遍历(二)

    C语言数据结构与算法之图的遍历(二)

    这篇文章主要是介绍了利用广度优先算法实现图的遍历,文中利用图文详细的介绍了实现步骤,对我们学习数据结构与算法有一定的帮助,需要的朋友可以参考一下
    2021-12-12
  • C++中std::vector的6种初始化方式

    C++中std::vector的6种初始化方式

    这篇文章主要介绍了C++中std::vector的6种初始化方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-08-08
  • 浅析C/C++中的可变参数与默认参数

    浅析C/C++中的可变参数与默认参数

    C支持可变参数的函数,这里的意思是C支持函数带有可变数量的参数,最常见的例子就是我们十分熟悉的printf()系列函数。我们还知道在函数调用时参数是自右向左压栈的
    2013-09-09
  • 详解C语言中的预处理命令

    详解C语言中的预处理命令

    初学C语言的时候,我们会在开头写下一句话,#include<stdio.h>,这就是预处理命令,下面我们通过这篇文章来了解一下,感兴趣的可以跟随小编一起学习一下
    2022-12-12
  • C++内存管理面经

    C++内存管理面经

    这篇文章主要介绍了C++的内存分配方式以及介绍了下栈和堆的区别,感兴趣的小伙伴可以参考阅读本文
    2023-03-03
  • 成员函数的重载、覆盖与隐藏详细解析

    成员函数的重载、覆盖与隐藏详细解析

    成员函数的重载、覆盖(override)与隐藏很容易混淆,C++程序员必须要搞清楚概念,否则错误将防不胜防
    2013-10-10
  • C++实现单例模式的自动释放

    C++实现单例模式的自动释放

    这篇文章主要为大家详细介绍了C++实现单例模式的自动释放,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-06-06
  • 解析内存对齐 Data alignment: Straighten up and fly right的详解

    解析内存对齐 Data alignment: Straighten up and fly right的详解

    对于所有直接操作内存的程序员来说,数据对齐都是很重要的问题.数据对齐对你的程序的表现甚至能否正常运行都会产生影响
    2013-05-05
  • C语言学生管理系统源码分享

    C语言学生管理系统源码分享

    这篇文章主要为大家分享了C语言学生管理系统的源码,帮助大家学习结构体,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-03-03
  • C++数据结构之搜索二叉树的实现

    C++数据结构之搜索二叉树的实现

    了解搜索二叉树是为了STL中的map和set做铺垫,我们所熟知的AVL树和平衡搜索二叉树也需要搜索二叉树的基础。本文将详解如何利用C++实现搜索二叉树,需要的可以参考一下
    2022-05-05

最新评论