C++获取时间函数众多,何时该用什么函数,拿到的是什么时间?该怎么用?很多人都会混淆。

本文是本人经历了几款游戏客户端和服务器开发后,对游戏中时间获取的一点总结。

最早学习游戏客户端时,为了获取最精确的时间,使用两个函数

BOOL QueryPerformanceFrequency(LARGE_INTEGER *lpFrequency);

BOOL QueryPerformanceCounter(LARGE_INTEGER *lpCount);

这两个函数分别是获取CPU的时钟频率和CPU计数器,是能够获取到的最精确的时间差。对于需要获取每帧走过的精确时间,使用这两个函数是最最精确的。

#include<windows.h>

LARGE_INTEGER lFrequency;
QueryPerformanceFrequency(&lFrequency); LARGE_INTEGER lBeginCount;
QueryPerformanceCounter(&lBeginCount); Sleep(); LARGE_INTEGER lEndCount;
QueryPerformanceCounter(&lEndCount); double time = (double)(lEndCount.QuadPart - lBeginCount.QuadPart) / (double)lFrequency.QuadPart;

在客户端代码的时间处理模块中,每一帧调用QueryPerformanceCounter获取当前的counter,即可获取每一帧使用的时间。(当然现在有Unity,估计没人关注这俩函数了)

虽然利用这两个函数能够精确的统计经过的时间,但是却无法得到当前时间,并且以上两个函数是Windows系统所特有的,unix/linux系统中并不具备。

为了获取系统的当前精确时间,需要使用另一个系统函数

int gettimeofday(struct timeval *tv, struct timezone *tz);

获取从1970年1月1日到现在经过的时间和时区(UTC时间),(按照linux的官方文档,时区已经不再使用,正常应该传NULL)。

#include <sys/time.h>

struct timeval start_tv;

gettimeofday(&start_tv, NULL);

sleep();

struct timeval end_tv;

gettimeofday(&end_tv, NULL);

double time = (end_tv.tv_sec - start_tv.tv_sec) + (double)(end_tv.tv_usec - start_tv.tv_usec)/(double);

这样同样可以获得精确到微秒的每帧经过的时间。服务器上的帧运行机制,一般便是这个时间函数来计算和同步。

如果不需要非常精确的时间,而只要精确到秒,可以使用另一个时间函数

time_t time(time_t* timer);

该函数返回一个UTC时间戳,如果传入timer参数,则为timer设置时间戳的值。

然而以上两个函数获取的都是UTC时间戳,如果在游戏中需要显示当前时区的时间,该怎么办呢?

使用localtime或localtime_r,两者效果一致,只是获取结果参数位置不同。

struct tm *localtime(const time_t *timep); // 传入UTC时间戳,返回当前时区的tm结构指针

struct tm *localtime_r(const time_t *timep, struct tm *result);

第一个函数获取tm结构静态变量的指针,第二个函数则传入一个tm结构变量的地址,并为之赋值。最终得到的tm变量,存储了当前时区的时间。

tm的结构定义如下,可以直接用来显示。

struct tm {
int tm_sec; /* seconds */
int tm_min; /* minutes */
int tm_hour; /* hours */
int tm_mday; /* day of the month */
int tm_mon; /* month */
int tm_year; /* year */
int tm_wday; /* day of the week */
int tm_yday; /* day in the year */
int tm_isdst; /* daylight saving time */
};

通常来说服务器和客户端通信同步时间时,不会传这么多int,只会传一个int64的UTC时间戳,给客户端自己转成当前时区。另外服务器也不会每次都调一次localtime转一次。一般来说,服务器和客户端都是维护一个int64的当前时间的UTC时间戳,以及一个当前时区的偏移时间,(客户端还会定时更新和服务器的时间误差,对时)。

因此,需要用到另外两个函数:

struct tm *gmtime(const time_t *timep); // time_t 到 tm 的转换,前后都是UTC时间,没有时区转换

struct tm *gmtime_r(const time_t *timep, struct tm *result); // gmtime 传入result方式

time_t mktime(struct tm *timeptr); // localtime的逆向操作,从当前时区的 tm 转到UTC的 time_t

在进程启动时

time_t t0 = ; // 当前时间为0时

time_t t1 = mktime(gmtime(&t0)); // UTC的时间

int timezone_diff = (int)(t0 - t1); // 当前时区和UTC的时间差

每次需要拿到当前时区时间,只需要用 UTCStamp + timezone_diff 即可。

除了获取时间,为了格式化显示时间,我们也可以利用一些系统函数,格式化输出时间字符串

char *asctime(const struct tm *tm);

char *asctime_r(const struct tm *tm, char *buf);

char ctime(const time_t *timep);

char ctime_r(const time_t *timep, char *buf);

在游戏的实际应用中,一般都是根据具体需求来显示格式化的时间,甚至要做一些边界时间的特殊处理,因此这里都不详细讨论时间的格式化显示了。

C++中的时间函数的更多相关文章

  1. 借助JavaScript中的时间函数改变Html中Table边框的颜色

    借助JavaScript中的时间函数改变Html中Table边框的颜色 <html> <head> <meta http-equiv="Content-Type ...

  2. ylb:SQL Server中的时间函数

    ylbtech-SQL Server:SQL Server-SQL Server中的时间函数 SQL Server中的时间函数. 1,SQL Server中的时间函数 返回顶部 1.   当前系统日期 ...

  3. C中的时间函数的用法

    C中的时间函数的用法    这个类展示了C语言中的时间函数的常用的用法. 源代码: #include <ctime>#include <iostream> using name ...

  4. SQLSERVER数据库中的 时间函数

    一.sql server日期时间函数 Sql Server中的日期与时间函数 1.  当前系统日期.时间 select getdate() 2. dateadd  在向指定日期加上一段时间的基础上,返 ...

  5. PHP中日期时间函数date()用法总结

    date()是我们常用的一个日期时间函数,下面我来总结一下关于date()函数的各种形式的用法,有需要学习的朋友可参考. 格式化日期date() 函数的第一个参数规定了如何格式化日期/时间.它使用字母 ...

  6. 【推荐】PHP中格式化时间函数date与gmdate的区别 | 修改PHP的默认时区

    PHP中的时间有2个格式化函数:date()和gmdate(),在官方的文档中的描述为: date -- 格式化一个本地时间/日期 gmdate -- 格式化一个 GMT/UTC 日期/时间,返回的是 ...

  7. Linux C++中的时间函数(转)

    http://net.pku.edu.cn/~yhf/linux_c/function/03.html   asctime(将时间和日期以字符串格式表示) 相关函数 time,ctime,gmtime ...

  8. FineReport中日期时间函数使用总结

    说明:凡函数中以日期作为参数因子的,其中日期的形式都必须是yy/mm/dd.而且必须用英文环境下双引号(" ")引用. DATE DATE(year,month,day):返回一个 ...

  9. Oracle中的时间函数用法(to_date、to_char) (总结)

    一.24小时的形式显示出来要用HH24 select to_char(sysdate,'yyyy-MM-dd HH24:mi:ss') from dual; select to_date('2005- ...

随机推荐

  1. mapreduce中一个map多个输入路径

    package duogemap; import java.io.IOException; import java.util.ArrayList; import java.util.List; imp ...

  2. 浅谈 Fragment 生命周期

    版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 源码:AndroidDemo/Fragment 文中如有纰漏,欢迎大家留言指出. Fragment 是在 Android 3.0 中 ...

  3. 从0开始搭建SQL Server AlwaysOn 第一篇(配置域控)

    从0开始搭建SQL Server AlwaysOn 第一篇(配置域控) 第一篇http://www.cnblogs.com/lyhabc/p/4678330.html第二篇http://www.cnb ...

  4. 前端学HTTP之日志记录

    前面的话 几乎所有的服务器和代理都会记录下它们所处理的HTTP事务摘要.这么做出于一系列的原因:跟踪使用情况.安全性.计费.错误检测等等.本文将谥介绍日志记录 记录内容 大多数情况下,日志的记录出于两 ...

  5. Android性能优化之利用Rxlifecycle解决RxJava内存泄漏

    前言: 其实RxJava引起的内存泄漏是我无意中发现了,本来是想了解Retrofit与RxJava相结合中是如何通过适配器模式解决的,结果却发现了RxJava是会引起内存泄漏的,所有想着查找一下资料学 ...

  6. Linux上如何查看物理CPU个数,核数,线程数

    首先,看看什么是超线程概念 超线程技术就是利用特殊的硬件指令,把两个逻辑内核模拟成两个物理芯片,让单个处理器都能使用线程级并行计算,进而兼容多线程操作系统和软件,减少了CPU的闲置时间,提高的CPU的 ...

  7. 谈一谈NOSQL的应用,Redis/Mongo

    1.心路历程 上年11月份来公司了,和另外一个同事一起,做了公司一个移动项目的微信公众号,然后为了推广微信公众号,策划那边需要我们做一些活动,包括抽奖,投票.最开始是没有用过redis的,公司因为考虑 ...

  8. C++随笔:.NET CoreCLR之GC探索(3)

    有几天没写GC相关的文章了哈,今天我讲GC的方式是通过一个小的Sample来讲解,这个小的示例代码只有全部Build成功了才会有.地址为D:\coreclr2\coreclr\bin\obj\Wind ...

  9. Solr高级查询Facet

    一.什么是facet solr种以导航为目的的查询结果成为facet,在用户查询的结果上根据分类增加了count信息,然后用户根据count信息做进一步搜索. facet主要用于导航实现渐进式精确搜索 ...

  10. 个人网站对xss跨站脚本攻击(重点是富文本编辑器情况)和sql注入攻击的防范

    昨天本博客受到了xss跨站脚本注入攻击,3分钟攻陷--其实攻击者进攻的手法很简单,没啥技术含量.只能感叹自己之前竟然完全没防范. 这是数据库里留下的一些记录.最后那人弄了一个无限循环弹出框的脚本,估计 ...