C++如何实现简单的计时器详解

实现分析

首先我们先分析一下计时器的一些功能,简单一点的计时器包括开始、暂停、停止和显示基本功能,这些功能以C++面向对象的编程思想(OOP)进行抽象,就是计时器类(Timer)的4个成员函数,当然我们要把这些函数作为公有的,因为它们是留给外部的接口(interface)。

然后我们再分析一下计时器的三种状态:停止,正在运行,暂停(注意:暂停不是停止),那么怎么记录计时器的三种状态呢?

这里我们用布尔类型的变量记录计时器的三种状态,分别为bool is_pause,bool is_stop,在这里一定要注意变量的命名,就像这样见名知意。吐舌头当然为了体现C++类的封装性,要把这两个bool变量作为计时器类(Timer)的私有成员。

实现方法

重要的问题来了,我们怎么实现计时呢?呐,你们知道time()函数吗?简单的说一下time函数:这个函数在time.h头文件中,它返回自 Unix 纪元(January 1 1970 00:00:00 GMT)起的当前时间的秒数,是一个长整形(long)。这样我们在计时器开始的时候获取一个time函数的返回值,保存到一个变量中,在用一个变量保存暂停时的时间。当我们开始计时的时候,进入一个死循环,始终用time() -开始的时间,就是当前计时器的时间。(不要急,具体实现往下看)。此时为了保存开始和暂停时刻的时间,必须在计时器类(Timer)中增加两个长整形(long)变量:start_time,pause_time,当然作为私有变量。

好像还少点什么......为了让外部获取当前计时器的状态,我们还需要两个函数返回计时器的状态,is_pause()计时器是否处于暂停,is_stop()计时器是否处于停止状态,返回值的类型为bool。

惊讶别忘了Timer的构造函数,用来做一些初始化的工作!

好了,至此我们的计时器类设计完成了(prefect),代码在下面:

class Timer
{
 private:
 long start_time;
 long pause_time;
 //两个bool值标记四种状态
 bool is_pause; //记录计时器的状态 (是否处于暂停状态)
 bool is_stop;//是否处于停止状态
 public:
 Timer();
 bool isPause(); //返回计时器状态
 bool isStop();
 //计时器的三种动作(功能)
 void Start();
 void Pause();
 void Stop();
 inline long getStartTime() {return start_time;}
 void show();
}; 

接下来的任务就是实现Timer的成员函数了..............

首先是构造函数Timer::Timer() ,完成一些初始化的工作:

Timer::Timer()
{
 is_pause = false; //初始化计时器状态
 is_stop = true;
} 

计时器开始计时之前应该处于停止状态!(计时器只能处于一种状态,不要犯糊涂哦!)

成员函数isPause()它的作用仅是让外部获取计时器是否处于暂停状态,so easy

bool Timer::isPause()
{
 if(is_pause)
 return true;
 else
 return false;
} 

isStop函数和isPause一样只是一个接口,让外部获取计时器的状态:

bool Timer::isStop() 

    if(is_stop) 
        return true; 
    return false; 
}

下面是Timer::Start()函数的实现,它是计时器开始的时候要运行的函数:

void Timer::Start() //开始
{
 if(is_stop)
 {
 start_time = time(0);
 is_stop = false;
 }
 else if(is_pause)
 {
 is_pause = false;
 start_time += time(0)-pause_time; //更新开始时间:用此时的时间 - 暂停时所用的时间 + 上一次开始的时间 = 此时的开始时间
 }
} 

我们首先判断一下计时的状态,如果处于停止状态,获取开始的时间,然后更新计时器的状态;如果计时器正处在暂停状态,我们让计时器继续计时,我采用改变开始的计时的时间(start_time)去调整计时的时间 : (用此时的时间 - 暂停时所用的时间 + 上一次开始的时间 = 此时的开始时间)。如果计时器正处于运行状态,就什么也不做!(不知道大家能不能看的懂.....)

这是暂停函数Timer::Pause()的实现:

void Timer::Pause() //暂停
{
 if(is_stop||is_pause) //如果处于停止/暂停状态,此动作不做任何处理,直接返回
 return;
 else //否则调制为暂停状态
 {
 is_pause = true;
 pause_time = time(0); //获取暂停时间
 }
} 

如果没有在运行,也就是处于暂停或停止状态,什么也不做直接返回。否则就去处理暂停请求:既然我们进行了暂停的操作,就要改变计时器的状态,将状态设置为暂停,保存此刻的时间,(这个暂停的时间pause_time在上面的开始的函数中用的到!)。

接着我们去实现停止函数Timer::Stop():

void Timer::Stop() //停止
{
 if(is_stop) //如果正处于停止状态(不是暂停状态),不做任何处理
 return ;
 else if(is_pause) //改变计时器状态
 {
 is_pause = false;
 is_stop = true;
 }
 else if(!is_stop)
 {
 is_stop = true;
 }
} 

如果处于停止状态,直接返回。否则如果处于暂停状态改变计时器的状态为is_stop = true ;否则就是处于运行状态,直接改变计时器的状态为停止。

下面是显示时间的函数Timer::show()

void Timer::show()
{
 long t = time(0) - start_time;
 gotoxy(35,12);
 cout<<setw(2)<<setfill('0')<<t/60/60<<":"
 <<setw(2)<<setfill('0')<<t/60<<":"
 <<setw(2)<<setfill('0')<<t%60<<endl;
} 

这里我要说一下gotoxy(int x,int y)函数,它的作用是将控制台的光标定位到坐标(x,y)处,show函数是要放到死循环中的,所以这样始终将输出的时间打印到一个地方,实现了时间的更新(我是不是很聪明);setw(int x)是设置输出的字宽,setfill(char  ch)设置了字符的填充。time函数返回的是秒数,t/60/60得到小时,t/60得到分钟,t%60得到秒数。

下面就是主函数了main()

int main()
{
 Timer t;
 char ch;
 hidden();//隐藏光标
 system("color 02");
 gotoxy(35,12);
 cout<<"00:00:00";
 gotoxy(20,18);
 cout<<"按a开始,按空格暂停,按s停止";
 while(1)
 {
 if(kbhit())
 {
  ch = getch();
  switch (ch)
  {
  case 'a':t.Start();break;
  case 's':t.Stop();break;
  case ' ':t.Pause();break;
  default :break;
  }
 }
 if(!t.isStop()&&!t.isPause())
 {
  t.show();
 }
 } 

} 

Timer t;定义一个计时器。hidden() ;是用来隐藏控制台光标的,不是必须的。

然后是进入死循环,kbhit()函数是检测是否有按键,如有按键返回非0值,没有按键返回0;用getch()获取按键,然后用

switch case分支结构处理不同的按键。

至此,我们的计时器设计完成!是不是感觉很简单啊!吐舌头

让我们看一下运行结果:

下面是本程序用到的头文件:

#include <iostream>
#include <cstdlib>
#include <ctime>
#include <conio.h>
#include <iomanip>
#include <windows.h> 

下面是代码中用到的函数:

void gotoxy(int x,int y)

void gotoxy(int x, int y)//定位光标,x为行坐标,y为列坐标
{
 COORD pos = {x,y};//(坐标 位置);
 HANDLE hOut = GetStdHandle(STD_OUTPUT_HANDLE); //得到标准处理(标准输出处理);
 SetConsoleCursorPosition(hOut, pos);//设置控制台光标位置;
} 

void hidden( )

void hidden()//隐藏光标
{
 HANDLE hOut = GetStdHandle(STD_OUTPUT_HANDLE);
 CONSOLE_CURSOR_INFO cci;
 GetConsoleCursorInfo(hOut,&cci);
 cci.bVisible=0;//赋1为显示,赋0为隐藏
 SetConsoleCursorInfo(hOut,&cci);
} 

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者能带来一定的帮助,如果有疑问大家可以留言交流。

(0)

相关推荐

  • C++实现统计代码运行时间计时器的简单实例

     C++实现统计代码运行时间计时器的简单实例 一.前言 这里记下从网上找到的一些自己比较常用的C++计时代码 二.Linux下精确至毫秒 #include <sys/time.h> #include <iostream> #include <time.h> double get_wall_time() { struct timeval time ; if (gettimeofday(&time,NULL)){ return 0; } return (double

  • C++如何实现简单的计时器详解

    实现分析 首先我们先分析一下计时器的一些功能,简单一点的计时器包括开始.暂停.停止和显示基本功能,这些功能以C++面向对象的编程思想(OOP)进行抽象,就是计时器类(Timer)的4个成员函数,当然我们要把这些函数作为公有的,因为它们是留给外部的接口(interface). 然后我们再分析一下计时器的三种状态:停止,正在运行,暂停(注意:暂停不是停止),那么怎么记录计时器的三种状态呢? 这里我们用布尔类型的变量记录计时器的三种状态,分别为bool is_pause,bool is_stop,在这

  • C++ 简单的任务队列详解

    任务队列是指能够实现任务在多线程间安全传递的先入先出的队列. 任务是指组合了数据和操作的对象,这里面定义为CTask类的对象. 任务的实现: Task.cpp #include "stdafx.h" #include "Task.h" #include <iostream> using namespace std; CTask::CTask(int* nCount) { m_nCount = nCount; } CTask::~CTask() { } v

  • C# 中SharpMap的简单使用实例详解

    本文是利用ShapMap实现GIS的简单应用的小例子,以供学习分享使用.关于SharpMap的说明,网上大多是以ShapeFile为例进行简单的说明,就连官网上的例子也不多.本文是自己参考了源代码进行整理的,主要是WinForm的例子.原理方面本文也不过多论述,主要是实例演示,需要的朋友还是以SharpMap源码进行深入研究. 什么是SharpMap ? SharpMap是一个基于.net 2.0使用C#开发的Map渲染类库,可以渲染各类GIS数据(目前支持ESRI Shape和PostGIS格

  • CentOS安装mysql5.7 及简单配置教程详解

    安装 保证你的用户有权限 安装 没有 切换 root su root (su的意思:swich user) # rpm -ivh http://dev.mysql.com/get/mysql57-community-release-el6-9.noarch.rpm 可能会遇到 warning: /var/tmp/rpm-tmp.6V5aFC: Header V3 DSA/SHA1 Signature, key ID 5072e1f5: NOKEY 可以忽略(个人意见,百度了一下没找到合适的答案)

  • Android 实现夜间模式的快速简单方法实例详解

    ChangeMode 项目地址:ChangeMode Implementation of night mode for Android. 用最简单的方式实现夜间模式,支持ListView.RecyclerView. Preview Usage xml android:background="?attr/zzbackground" app:backgroundAttr="zzbackground"//如果当前页面要立即刷新,这里传入属性名称 比如 R.attr.zzb

  • Linux上安装Mysql及简单的使用详解

    1. 安装mysql sudo apt-get update sudo apt-get install mysql-server sudo apt-get install python-mysqldb(如果python中要使用,请安装) 2.登录root用户: mysql -u root -p 3.查看所有数据库: show databases; 4.选择一个数据库操作: use database_name; 5.查看当前数据库下所有的表: show tables; 6.创建一个数据库: cre

  • IOS文件的简单读写实例详解

    IOS文件的简单读写实例详解 数组(可变与不可变)和字典(可变与不可变)中元素对象的类型,必须是NSString,NSArray,NSDictionary,NSData,否则不能直接写入文件 #pragma mark---NSString的写入与读取--- //1:获取路径 NSString *docunments = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)firstObje

  • 微信小程序之网络请求简单封装实例详解

    微信小程序之网络请求简单封装实例详解 在微信小程序中实现网络请求相对于Android来说感觉简单很多,我们只需要使用其提供的API就可以解决网络请求问题. 普通HTTPS请求(wx.request) 上传文件(wx.uploadFile) 下载文件(wx.downloadFile) WebSocket通信(wx.connectSocket) 为了数据安全,微信小程序网络请求只支持https,当然各个参数的含义就不在细说,不熟悉的话可以:可以去阅读官方文档的网络请求api,当我们使用request

  • Spring Boot的listener(监听器)简单使用实例详解

    监听器(Listener)的注册方法和 Servlet 一样,有两种方式:代码注册或者注解注册 1.代码注册方式 通过代码方式注入过滤器 @Bean public ServletListenerRegistrationBean servletListenerRegistrationBean(){ ServletListenerRegistrationBean servletListenerRegistrationBean = new ServletListenerRegistrationBean

  • Kotlin + Retrofit + RxJava简单封装使用详解

    本文介绍了Kotlin + Retrofit + RxJava简单封装使用详解,分享给大家,具体如下: 实例化Retrofit object RetrofitUtil { val CONNECT_TIME_OUT = 30//连接超时时长x秒 val READ_TIME_OUT = 30//读数据超时时长x秒 val WRITE_TIME_OUT = 30//写数据接超时时长x秒 val retrofit: Retrofit by lazy { Log.d("RetrofitUtil"

随机推荐