C言語で信頼できる時間誤差
C言語のQueryPerformanceCounterを用いて時間を計測するプログラムを作りました.
このとき時間の間隔は必ずしも500m秒になりません.
原因としては,
1. Sleep関数に含まれる誤差
2. QueryPerformanceCounter関数に含まれる誤差
の2つが考えられると思っています.
そこで,質問ですが,1および2の誤差ってどの程度なのでしょうか?
また,その他に誤差が出現するとすれば,それは何でしょうか.
ちなみに,PCはwindows XPでMicrosoft Viusal Stadio 2010で動作させ,
その他のアプリケーションは開いていない状況下で実行しました.
サンプルプログラムにより実行しましたが,何が原因なのでしょうか?
**********【サンプルプログラム】**********
#include <stdio.h>
#include <windows.h>
void main (void){
LARGE_INTEGERm_pCounterFreq;
LARGE_INTEGERm_pBeforeCounter;
LARGE_INTEGER ulCounter;
double dPassageSec;
double pre =0.0;
int counter;
printf("取得時間,取得時間間隔\n");
QueryPerformanceFrequency( &m_pCounterFreq );
QueryPerformanceCounter( &m_pBeforeCounter );
for(counter=0;;counter++){
QueryPerformanceCounter( &ulCounter );
dPassageSec = (double)(ulCounter.QuadPart - m_pBeforeCounter.QuadPart) / (double)m_pCounterFreq.QuadPart;
if(dPassageSec >= 5.2) break;
Sleep(500);
printf("%-12f,%-12f\n", dPassageSec, dPassageSec-pre);
pre = dPassageSec;
}
}
**********【実行結果一例】**********
取得時間,取得時間間隔
0.000002 ,0.000002
0.486344 ,0.486342←14msも誤差がある
0.986095 ,0.499751
1.486106 ,0.500012
1.986102 ,0.499995
2.486100 ,0.499998
2.986116 ,0.500016
3.486089 ,0.499973
3.986090 ,0.500001←ほぼ500ms
4.486092 ,0.500002
4.986208 ,0.500117
No.1ベストアンサー
- 回答日時:
たぶん間違いないと思いますが、MS-WindowsもUNIXやLinuxもリアルタイムOSではありません。
ご存じのように、これらのOSはマルチタスクにより、同時にいろいろな処理が動いています。
そして、Sleepすると実行権を解放します。
その後、Sleep終了後に実行権が与えられるのですが(この場合であれば)500ms後丁度に実行権が与えられることは保証されていません。
確か、NASAはLinuxを使用しているはずですが、これはLinuxをリアルタイムOSに改造したものを使用していると聞きました。
ロケットの打ち上げでは、チョットの時間のズレも一大事になる可能性がありますからね。
Sleep誤差ですか.参考になりました.
NASAのリアルタイムOSは気になりますね!
少し調べてみます.ありがとうございます,
No.6
- 回答日時:
Sleep関数の精度についてはMSDNの英語のページに説明があります。
日本語にはありませんが。http://msdn.microsoft.com/en-us/library/ms686298 …
http://msdn.microsoft.com/ja-jp/library/cc429358 …
ここも参考になります。
http://hp.vector.co.jp/authors/VA007219/rtc_pic. …
結論としてはSleepは高精度のタイマーとしては使えないと思います。
No.5
- 回答日時:
Sleepですね。
Sleepの動作としてはプロセスの実行権を手放して指定時間を過ぎてから実行可能状態にするだけです。指定時間を過ぎてもカーネル、デバイスドライバなど特権プログラムの実行中だと実行可能状態になるまでに時間が掛かる場合もありますし、実行可能状態になっても他の実行可能プロセスとの競合で実行再開が遅れる場合があります。従って、再開時間の精度を上げようと思うと以下が必要になります。
1. カーネル、デバイスドライバ等の特権プログラムでの割り込み禁止時間を短くする
2. プロセスを他のプロセスより優先度を高くする
このうち2はプロセス起動時の設定などでできますが、1は使う全てのデバイスに関わるため簡単ではありません。
数年前にLinuxで調べたときにはディスク・デバイスドライバに長い割り込み禁止があり時々タイマに数百msを越える遅延がありました。
なお、QueryPerformanceCounterも結構重いですが、待ちは伴わないので数マイクロ秒程度の遅延しかありませんし、取得する値にはほぼ誤差はないはずです。
このプログラムは予備段階で他に色々とデバイスを読み込んだりするので,
windowsの方が都合がいいので,OS変更はできないんです.
でも,OSによる違いが生まれるという知識は今後大いに役立つと思います.
ありがとうございました.
No.4
- 回答日時:
>1. Sleep関数に含まれる誤差
>2. QueryPerformanceCounter関数に含まれる誤差
QueryPerformanceCounterの値は、CPUのクロックに使われている水晶発振子からきているので、誤差はほぼゼロと考えてよいです。
この場合の原因はSleepの誤差ですね。
理由は他の方もおっしゃっている通りで、Sleepとはそもそもそんなものと考えて使うべきでしょう。
なるべく正確に500msウェイトしたいなら、Sleepで490msとか480msとか待つようにして、最終調整にQueryPerformanceCounterを使うのがいいと思います。
まあ、それでも他のアプリとの競合次第では、大幅にタイミングが遅れる可能性はあります。
そのあたりをどの程度まで許容できるかですね。
絶対に500msを外せないのであれば、OSの選定から考えなければならないでしょう。
説明が詳しくてわかりやすくて助かります.
なるほど,CPUのクロックに用いられているものなら,間違いないでしょうね.
0.1s以下は正直誤差があっても問題はないのですが,
質問されたときに正確に応えたかったもので...
参考になりました.ありがとうございます.
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
関連するカテゴリからQ&Aを探す
おすすめ情報
- ・漫画をレンタルでお得に読める!
- ・街中で見かけて「グッときた人」の思い出
- ・「一気に最後まで読んだ」本、教えて下さい!
- ・幼稚園時代「何組」でしたか?
- ・激凹みから立ち直る方法
- ・1つだけ過去を変えられるとしたら?
- ・【あるあるbot連動企画】あるあるbotに投稿したけど採用されなかったあるある募集
- ・【あるあるbot連動企画】フォロワー20万人のアカウントであなたのあるあるを披露してみませんか?
- ・映画のエンドロール観る派?観ない派?
- ・海外旅行から帰ってきたら、まず何を食べる?
- ・誕生日にもらった意外なもの
- ・天使と悪魔選手権
- ・ちょっと先の未来クイズ第2問
- ・【大喜利】【投稿~9/7】 ロボットの住む世界で流行ってる罰ゲームとは?
- ・推しミネラルウォーターはありますか?
- ・都道府県穴埋めゲーム
- ・この人頭いいなと思ったエピソード
- ・準・究極の選択
- ・ゆるやかでぃべーと タイムマシンを破壊すべきか。
- ・歩いた自慢大会
- ・許せない心理テスト
- ・字面がカッコいい英単語
- ・これ何て呼びますか Part2
- ・人生で一番思い出に残ってる靴
- ・ゆるやかでぃべーと すべての高校生はアルバイトをするべきだ。
- ・初めて自分の家と他人の家が違う、と意識した時
- ・単二電池
- ・チョコミントアイス
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
Visual Studio 「AnyCpu」について
-
excel vbaから実行するexe実行...
-
実行ファイルの動作について
-
VBで「ファイル名を指定して実...
-
エスケープ・シーケンスによる...
-
「読み込み違反」が起きたとき...
-
C言語で信頼できる時間誤差
-
JP1/File Transmission Server/...
-
matlabで作成したdllをVBAで使...
-
デバック時エラーなし、アプリ...
-
VS2008 の C++ で作成したEXEの...
-
C++初心者です。 debug asserti...
-
ACCESSランタイムでエラー
-
コマンドライン引数について
-
VB6.0 exe作成時に実行時エラー...
-
EXCELファイルの複数ダウンロー...
-
意味不明の実行時エラーで困っ...
-
VBAでArrayListを使う為の「msc...
-
Designer.vbは直接コードをいじ...
-
vba 時間の引き算 例えば 15:00...
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
「読み込み違反」が起きたとき...
-
Visual Studio 「AnyCpu」について
-
excel vbaから実行するexe実行...
-
Windows PowerShellでC言語を実...
-
C++初心者です。 debug asserti...
-
C#VB、exeに埋め込んだexeの実行
-
実行ファイル(.exe)が別のPC...
-
ACCESSランタイムでエラー
-
VB6.0 exe作成時に実行時エラー...
-
一通ずつ処理したい(アウトル...
-
外部exe呼び出しの方法 ShellEx...
-
Format 関数のバグ?
-
Eclipse CDT MinGWによるC言語
-
JP1/File Transmission Server/...
-
VBで「ファイル名を指定して実...
-
VB.NetのWinsockについて
-
VBにてDLLをデバックする方法。
-
VB2005 Vistaで印刷ができません
-
androidで.exeを実行できますか?
-
ActiveX → VB を起動するとCre...
おすすめ情報