几何尺寸与公差论坛

 找回密码
 注册
查看: 1864|回复: 8

一种取得Windows系统运行时间,且不同于GetTickCount的方法[问题点数:100分,结帖人

[复制链接]
发表于 2010-4-14 21:15:28 | 显示全部楼层 |阅读模式
http://topic.csdn.net/u/20090904 ... b-80f5bb3ad574.html
Windows系统的应用程序编程接口函数(API)都是使用C/C++语言编写的,VB中使用系统API函数需要改写声明。
GetiTickCount函数的含义是:取得自Windows系统启动以来到现在所经过的时间(单位:ms)。
在VB6.0中,API函数:
GetTickCount的声明如下:Private  Declare Function GetTickCount Lib "kernel32" () As  Long
在Windows中,该函数的原型是:DWORD  GetiTickCount

我们可以发现在VB系统下,Gettickcount返回的是Long类型,长4个字节,而在WIndows下,GetTickCount返回的是DWORD类型,长4个字节的无符号数据类型。DWORD类型的数据范围是:0  ~  2^32。
Long类型的数据范围是:-2147483648~2147483647,作为GetTickCount函数的返回值,只能是:0~2147483647,经过计算大约是:24.86天,也就是说,使用VB编制的系统在使用了24.86天以后,使用GetTickCount读回来的值就是负数了!具体原因请参考:http://topic.csdn.net/u/20090708/20/c2d2370d-d506-441e-b3c0-5aa3a6bcd566.html

最近我偶然间发现一个高人提供的一个方法,感觉能弥补VB中GetTickCount函数的不足。代码如下:
Windows系统的应用程序编程接口函数(API)都是使用C/C++语言编写的,VB中使用系统API函数需要改写声明。
GetiTickCount函数的含义是:取得自Windows系统启动以来到现在所经过的时间(单位:ms)。
在VB6.0中,API函数:
GetTickCount的声明如下:Private  Declare Function GetTickCount Lib "kernel32" () As  Long
在Windows中,该函数的原型是:DWORD  GetiTickCount

我们可以发现在VB系统下,Gettickcount返回的是Long类型,长4个字节,而在WIndows下,GetTickCount返回的是DWORD类型,长4个字节的无符号数据类型。DWORD类型的数据范围是:0  ~  2^32。
Long类型的数据范围是:-2147483648~2147483647,作为GetTickCount函数的返回值,只能是:0~2147483647,经过计算大约是:24.86天,也就是说,使用VB编制的系统在使用了24.86天以后,使用GetTickCount读回来的值就是负数了!具体原因请参考:http://topic.csdn.net/u/20090708/20/c2d2370d-d506-441e-b3c0-5aa3a6bcd566.html

最近我偶然间发现一个高人提供的一个方法,感觉能弥补VB中GetTickCount函数的不足。代码如下:
VB code Option Explicit
Private Declare Function osQueryPerformanceCounter Lib "kernel32" Alias "QueryPerformanceCounter" _
                                                (lpPerformanceCount
As Currency) As Long
Private Declare Function osQueryPerformanceFrequency Lib "kernel32" Alias "QueryPerformanceFrequency" _
                                                (lpFrequency
As Currency) As Long
Private Declare Function GetTickCount Lib "kernel32" () As Long
'%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
'
函数功能:计算Windows自启动以来所经历的时间(s)
'
返回类型:Double类型,你可以修改它,也可以返回整形。
'
'
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
Public Function Timer() As Double
   
Dim freq As Currency
   
Dim count As Currency
On Error GoTo errFun
    osQueryPerformanceFrequency freq        
'频率 相当于速度 v
    osQueryPerformanceCounter count         '已发脉冲数 相当于距离 s
    Timer = count / freq                    '计算时间 t=s/v
    Exit Function
errFun:
   
Timer = 0
End Function

Private Sub Form_Load()
    Text1.Text
= ""
    Text2.Text
= ""
    Timer1.Enabled
= True
    Timer1.Interval
= 1000
End Sub
'%%%%%%%%%%%%%%%%%%%%%%
'
过程功能:使用两个方法计算系统启动时间,实际测试,两者值有点误差。
'
'
%%%%%%%%%%%%%%%%%%%%%%%
Private Sub Timer1_Timer()
    Text1.Text
= Timer
    Text2.Text
= GetTickCount * 0.001
End Sub
 楼主| 发表于 2010-4-14 21:16:00 | 显示全部楼层

回复: 一种取得Windows系统运行时间,且不同于GetTickCount的方法[问题点数:100分,

这是高人的原文:


需要说明的一点是,如果你查阅API手册,你会发现API并不能识别Currency数据类型。在API手册中,osQueryPerformanceFrequencey要求的参数是一个Large_Integer的数据结构,该数据结构由两个Long类型的数据组成。Currency实际上也是按同样的方法来储存的。只不过在计算时被除以10000。因此你也可以写成:, (freq/10000) / (count/10000) = freq/count。关于上面的两个API函数,osQueryPerformanceFrequecy表示一秒内,计时器Tick了多少次;osQueryPerformanceCounter表示从开机到现在,计时器一共Tick了多少次。二者相除,就是从开机到现在过去了多少秒钟。
 楼主| 发表于 2010-4-14 21:16:14 | 显示全部楼层

回复: 一种取得Windows系统运行时间,且不同于GetTickCount的方法[问题点数:100分,

我觉得最精确的是靠cpu计数器配合误差补偿算法。
在《深入理解计算机系统》里面有介绍。
 楼主| 发表于 2010-4-14 21:17:02 | 显示全部楼层

回复: 一种取得Windows系统运行时间,且不同于GetTickCount的方法[问题点数:100分,

引用 13 楼 citybird 的回复:
计算机的脉冲估计是由晶体振荡器产生的确实不太精确,要想精确计时可以搞个原子钟啥的,几万年甚至几十万年的累计误差才1秒钟。


错。这个牵涉到天文学上很复杂的概念,简单地说是这样的:
物理上,我们要求一秒的长度是固定的。
生活中我们要求一天的长度是 24x3600 秒。
然而,很不幸的是,一天的长度并不固定,首先一天的定义应该是太阳两次通过子午线的间隔,然而地球的公转速度在一年内是不均匀的,冬天快夏天慢(开普勒定律)。就公转本身还存在进动和章动。
另外地球由于受到月球潮汐摩擦力的作用,自转速度会减慢,大约一个世纪几毫秒。
除此之外,还有不规则的变动,无法预计。
因此,每隔几年,我们就需要添加一个正闰秒或者负闰秒来消除这种误差。
所以时间只能由授时中心发布。
 楼主| 发表于 2010-4-14 21:18:02 | 显示全部楼层

回复: 一种取得Windows系统运行时间,且不同于GetTickCount的方法[问题点数:100分,

需要获取高精度的时间系统本身就提供了几个API呀,干嘛这么麻烦,去MSDN 查一下 QueryPerformanceCounter、QueryPerformanceFrequency
 楼主| 发表于 2010-4-14 21:18:17 | 显示全部楼层

回复: 一种取得Windows系统运行时间,且不同于GetTickCount的方法[问题点数:100分,

引用 60 楼 soft_dep 的回复:
需要获取高精度的时间系统本身就提供了几个API呀,干嘛这么麻烦,去MSDN 查一下 QueryPerformanceCounter、QueryPerformanceFrequency



直接用NtQuerySystemInformation不是更好
省却了函数之间的调用
 楼主| 发表于 2010-4-14 21:19:03 | 显示全部楼层

回复: 一种取得Windows系统运行时间,且不同于GetTickCount的方法[问题点数:100分,

高精度计数器,我已经玩过了。不过,陈辉说的NtQuerySystemInformation,还没有玩过,今天回去玩玩!
 楼主| 发表于 2010-4-14 21:19:53 | 显示全部楼层

回复: 一种取得Windows系统运行时间,且不同于GetTickCount的方法[问题点数:100分,

在多核情况下时间可能会计算得不准的,osQueryPerformanceCounter 取的是一个晶振震动的次数,
但是有可能会出现一个核与其它核的晶振震动次数不一致的情况,这时计算就会出现问题,具体原因MSDN
上有解释的,所以用这两个函数时,一定要将进行记时的线程绑定到某个CPU上,绑定的API函数是:SetThreadIdealProcessor,很多网游的服务器和客户端的时间同步模块都会有这样的规定的
 楼主| 发表于 2010-4-14 21:20:22 | 显示全部楼层

回复: 一种取得Windows系统运行时间,且不同于GetTickCount的方法[问题点数:100分,

引用 96 楼 sulipeng007 的回复:
long负数只是vb没有dword的概念,当long数据的4个字节共32位中的最高位为1时,vb就显示为负的,而且此时值是为该负数的补码,如果你需要dowrd值,那么在vb中定义一个能包含dword数据范围的数据类型,如currency,然后直接用“&H100000000+long负值”就可以得到dword表示的正值了。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

QQ|Archiver|小黑屋|几何尺寸与公差论坛

GMT+8, 2024-5-10 03:29 , Processed in 0.045525 second(s), 19 queries .

Powered by Discuz! X3.4 Licensed

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表