C语言小项目——计时器(倒计时+报警提示)「建议收藏」

大家对计时器应该不陌生,我们在制定一个计划时,经常喜欢设置一个倒计时来规定完成时限,等计时结束,它还会报警提示,今天,我就用C语言编写一个简易的倒计时计时器。

大家好,又见面了,我是你们的朋友全栈君。

大家对计时器应该不陌生,我们在制定一个计划时,经常喜欢设置一个倒计时来规定完成时限,等到计时结束,它还会报警提示,今天,我就用C语言编写一个简易的倒计时计时器。

实现思路

简单介绍一下我的实现思路:
本文包括三个版本,分别是极简版普通版高级版

  1. 极简版:简单的单次倒计时,只能设置秒数,计时通过Sleep()函数实现(粗略倒计时),先输入秒数cnt,然后每过1秒(Sleep(1000)),cnt减1,直到cnt变成0,就退出程序。
  2. 普通版:可设置计时的hour,min,sec),计时通过time()函数实现(用来获取当前系统时间的秒数),进入while循环,判断系统时间秒数是否变化,如有变化,说明过了1秒。这时将秒数sec减1,当sec减到0,下一秒就将分钟min减1,同时sec变为59。分钟min和小时hour的变化关系与秒和分钟的关系类似。一直循环倒计时,直到hour,minsec都变为0,此时计时结束,进行报警提示printf("\a"))。
  3. 高级版:在普通版的基础上,增加重复计时暂停计时的功能。首先需要设定计时时间,然后按下空格键开始计时,再次按下空格键,可以暂停倒计时,同时设置的倒计时时间具有记忆功能,即计时结束后,再次按下空格,依然按照上次设定的计时时间进行计时。

1.极简版本

极简版代码很简单,就不多作介绍了。

#include <stdio.h>
#include <Windows.h> //Sleep(ms)
int main()
{ 
   
    int cnt = 0;
    puts("输入秒数");
    scanf("%d", &cnt);
    while(--cnt >= 0)
    { 
   
        Sleep(1000);
        //printf("%d\r", cnt); //初始版本--存在BUG,见评论区
        printf("%-4d\r", cnt); //最大9999秒,
                               //如果要更高,把4加大即可
                               //-4表示左对齐,宽度固定为4字节
        //cnt--; //“cnt减1”放到while循环中(根据评论区建议进行优化)
    }
    printf("计时结束\n");
    return 0;
}

运行效果

请添加图片描述

2.普通版本

普通版本也很简单,主要包括:

  1. 设置计时时间
  2. 倒计时
  3. 蜂鸣器报警(系统提示音)
#include <stdio.h>
#include <time.h>

int main(void)
{ 
   
    int hour = 0, min = 0, sec = 0;
    time_t time_sec = 0;
    time_t old_sec = 0;
    printf("请设置计时时间——时 分 秒\n");
    scanf("%d%d%d", &hour, &min, &sec);
    time(&time_sec);              //获取当前秒数(1970-1-1 00:00:00到现在)
    printf("倒计时——%02d:%02d:%02d\r", hour, min, sec);
    old_sec = time_sec;           //更新旧的秒数
    while(hour > 0 || min > 0 || sec > 0)
    { 
   
        time(&time_sec);          //获取秒数保存到time_t变量
        if(time_sec != old_sec)   //如果秒数改变(计时达到1秒)
        { 
   
            old_sec = time_sec;   //更新旧的秒数
            if(sec > 0)
                sec--;            //计时秒数减1
            else
            { 
   
                sec = 59;         //如果原秒数为0,则变为59
                if(min > 0)
                    min--;        //计时分钟减1
                else
                { 
   
                    min = 59;     //如果分钟数为0,则变为59
                    hour--;       //计时小时数减1
                }
            }
            printf("倒计时——%02d:%02d:%02d\r", hour, min, sec);
        }
    }
    for(int i = 0; i < 5; i++)
    { 
   
        printf("\a");             //响蜂鸣器(或系统提示音)
        sleep(1);                 //延时1秒,太短电脑不发出声音
    }
    puts("\n计时结束\n");
    return 0;
}

运行效果

请添加图片描述

3.高级版本

在普通版本的基础上添加了菜单打印键盘按键扫描两部分代码。
程序的核心为 pause_flagmenu_flag这两个标志,前者控制倒计时的开始暂停继续,后者控制菜单是否需要刷新(只有程序第一次运行设置计时时间计时结束时才需要打印菜单)。

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

int main(void)
{ 
   
    //动态计时参数(打印值)
    unsigned int hour = 0, min = 0, sec = 0;
    //默认值(设定的计时参数)
    unsigned int hour_def = 0, min_def = 0, sec_def = 0;
    time_t time_sec = 0;
    time_t old_sec = 0;
    char key;
    int pause_flag = 1;            //暂停标志
    int menu_flag = 1;             //菜单打印标志
    while(1)
    { 
   
        /**************** 菜单打印 ****************/
        if(menu_flag == 1)
        { 
   
            menu_flag = 0;
            system("cls");          //Windows系统清屏命令
            printf("================================\n"); //菜单
            if(hour_def || min_def || sec_def)
                printf("|\t倒计时时间:%02d:%02d:%02d\t |\n",\
                       hour_def, min_def, sec_def);
            else
                printf("|\t未设置计时时间\t\t|\n");
            printf("| S:设置 空格:开始/暂停 Q:退出 |\n");
            printf("================================\n");
            if(pause_flag == 0)
            { 
   
                printf("计时结束\n");
                for(int i = 0; i < 5; i++)
                { 
   
                    printf("\a");             //响蜂鸣器(或系统提示音)
                    sleep(1);                 //延时1秒,太短电脑不发出声音
                }
                pause_flag = 1;          //停止计时

            }
        }

        /**************** 键盘按键扫描+操作 ****************/
        key = 0;
        if(_kbhit())                      //检测到按键按下
            key = getch();                //读取按键
        switch(key)
        { 
   
            case 's':                     //按s/S设置计时时间
            case 'S':
                printf("请设置计时时间——时 分 秒\n");
                scanf("%d %d %d", &hour_def, &min_def, &sec_def);
                if(hour_def > 24 || min_def > 59 || sec_def > 59)
                { 
   
                    //printf("时间设置失败\n");
                    hour_def = min_def = sec_def = 0; //重置时间
                }
                hour = min = sec = 0;       //计时值清零
                menu_flag = 1;              //打印菜单
                break;
            case ' ':                       //按空格键开始/暂停计时
                if(hour_def || min_def || sec_def)   //如果时间有效
                { 
   
                    if(pause_flag)
                    { 
   
                        pause_flag = 0;     //开始计时或继续计时
                        if(!hour && !min && !sec) //计数值为0
                        { 
   
                            hour = hour_def;   //读取上次设置的时间
                            min = min_def;
                            sec = sec_def;
                            //打印初始时间
                            printf("倒计时——%02d:%02d:%02d\r",\
                                   hour, min, sec);
                        }
                    }
                    else
                        pause_flag = 1;     //暂停计时
                    time(&time_sec);        //获取当前秒数(1970-1-1 00:00:00到现在)
                    old_sec = time_sec;     //更新旧的秒数
                }
                break;
            case 'q':
            case 'Q': printf("程序退出\n");return 0;
        }

        /**************** 计时操作 ****************/
        time(&time_sec);              //获取秒数保存到time_t变量
        if(pause_flag == 0 && old_sec != time_sec)
        { 
   
            old_sec = time_sec;       //更新旧的秒数
            if(sec > 0)
                sec--;                //计时秒数减1
            else
            { 
   
                sec = 59;             //如果原秒数为0,则变为59
                if(min > 0)
                    min--;            //计时分钟减1
                else
                { 
   
                    min = 59;         //如果分钟数为0,则变为59
                    hour--;           //计时小时数减1
                }
            }
            printf("倒计时——%02d:%02d:%02d\r", hour, min, sec);

            if(!hour && !min && !sec) //计时结束
                menu_flag = 1;        //打印菜单
        }
    }
    return 0;
}

运行效果

请添加图片描述

文章到这里就结束了,如果你有更好的方案,欢迎分享?,如果我程序有任何问题,可以评论或私聊我,一起学习,一起进步!!!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/126283.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • Java Web乱码分析及解决方式(一)——GET请求乱码

    Java Web乱码分析及解决方式(一)——GET请求乱码

    2022年2月7日
    38
  • java初中级面试题集锦

    java初中级面试题集锦目录基础&与&&区别使用final关键字修饰一个变量时,是引用不能变,还是引用的对象不能变?静态变量和实例变量的区别?是否可以从一个static方法内部发出对非static方法的调用?”==”和equals方法究竟有什么区别?int和Integer的区别什么是foreach循环,它可以循环那些数据类型重载与重写区别?…

    2022年6月18日
    29
  • 中文人物关系图谱构建与应用项目(人物关系抽取,关系抽取评测)

    中文人物关系图谱构建与应用项目(人物关系抽取,关系抽取评测)ChinesePersonRelationGraphChinesePersonRelationGraph,personrelationshipextractionbasedonnlpmethods.中文人物关系知识图谱项目,内容包括中文人物关系图谱构建,基于知识库的数据回标,基于远程监督与bootstrapping方法的人物关系抽取,基于知识图谱的知识问答等应用.项目地址:htt…

    2022年6月26日
    52
  • SpringBoot2.x系列教程汇总-从入门到精通「建议收藏」

    SpringBoot2.x系列教程汇总-从入门到精通「建议收藏」因为CSDN没有分类归纳博客的功能,所以特写本帖汇总SpringBoot2.x系列教程,方便大家查阅!SpringBoot2.x系列教程01–史前文明之Spring简介SpringBoot2.x系列教程02–史前文明之Spring发展史SpringBoot2.x系列教程03–新纪元之SpringBoot初相见SpringBoot系列教程04–新纪元之SpringBoot环境……

    2022年9月25日
    2
  • 区块链:P2P技术是什么

    区块链:P2P技术是什么摘要:包括比特币、以太坊等在内的去中心化的区块链平台,其底层网络都是采用的P2P技术实现,每个节点都是对等的。而本文,则先通过介绍P2P技术的特点和发展历史,让大家对P2P这个技术的来龙去脉有一个初步的认识和了解。然后在下一篇文章中,我会详细介绍第三代P2P技术(DHT)—Kademlia算法的实现原理。1.p2p的定义P2P是peer-to-peer的简称,又称为点对点技…

    2022年6月19日
    26
  • LeetCode重建二叉树详解[通俗易懂]

    LeetCode重建二叉树详解[通俗易懂]LeetCode重建二叉树详解题目描述原理分析(1)大致思路(2)细节阐述代码实现(1)主函数(2)递归函数参数区间的决定递归结束的条件总结题目描述原理分析(1)大致思路下面讲解一下,前序遍历+中序遍历如何确定一个唯一的二叉树。关于二叉树的基本知识,请看二叉树的基本操作及联系。对此就不再过多重复。对于前序遍历顺序:根、左子树、右子树;对于中序的遍历顺序:左子树、根、右子树。所以通过前序遍历,我们获取前序第一个结点就是这个树的根,再在中序遍历中找到该结点的位置。在中序中,根的左边全部的是属于根左子

    2022年6月12日
    37

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号