- ベストアンサー
C言語で信頼できる時間誤差
- C言語のQueryPerformanceCounterを用いて時間を計測するプログラムを作りました.このとき時間の間隔は必ずしも500m秒になりません.Sleep関数に含まれる誤差とQueryPerformanceCounter関数に含まれる誤差の2つが考えられると思っています.
- PCはwindows XPでMicrosoft Viusal Stadio 2010で動作させ,その他のアプリケーションは開いていない状況下で実行しました.
- サンプルプログラムにより実行しましたが,何が原因なのでしょうか?
- みんなの回答 (6)
- 専門家の回答
質問者が選んだベストアンサー
- ベストアンサー
たぶん間違いないと思いますが、MS-WindowsもUNIXやLinuxもリアルタイムOSではありません。 ご存じのように、これらのOSはマルチタスクにより、同時にいろいろな処理が動いています。 そして、Sleepすると実行権を解放します。 その後、Sleep終了後に実行権が与えられるのですが(この場合であれば)500ms後丁度に実行権が与えられることは保証されていません。 確か、NASAはLinuxを使用しているはずですが、これはLinuxをリアルタイムOSに改造したものを使用していると聞きました。 ロケットの打ち上げでは、チョットの時間のズレも一大事になる可能性がありますからね。
その他の回答 (5)
- goma1109
- ベストアンサー率0% (0/2)
Sleep関数の精度についてはMSDNの英語のページに説明があります。 日本語にはありませんが。 http://msdn.microsoft.com/en-us/library/ms686298%28VS.85%29.aspx http://msdn.microsoft.com/ja-jp/library/cc429358.aspx ここも参考になります。 http://hp.vector.co.jp/authors/VA007219/rtc_pic.html 結論としてはSleepは高精度のタイマーとしては使えないと思います。
お礼
英語の文献は少し抵抗があったのですが,参考にさせていただきます. ありがとうございます.
- rinkun
- ベストアンサー率44% (706/1571)
Sleepですね。Sleepの動作としてはプロセスの実行権を手放して指定時間を過ぎてから実行可能状態にするだけです。指定時間を過ぎてもカーネル、デバイスドライバなど特権プログラムの実行中だと実行可能状態になるまでに時間が掛かる場合もありますし、実行可能状態になっても他の実行可能プロセスとの競合で実行再開が遅れる場合があります。 従って、再開時間の精度を上げようと思うと以下が必要になります。 1. カーネル、デバイスドライバ等の特権プログラムでの割り込み禁止時間を短くする 2. プロセスを他のプロセスより優先度を高くする このうち2はプロセス起動時の設定などでできますが、1は使う全てのデバイスに関わるため簡単ではありません。 数年前にLinuxで調べたときにはディスク・デバイスドライバに長い割り込み禁止があり時々タイマに数百msを越える遅延がありました。 なお、QueryPerformanceCounterも結構重いですが、待ちは伴わないので数マイクロ秒程度の遅延しかありませんし、取得する値にはほぼ誤差はないはずです。
お礼
このプログラムは予備段階で他に色々とデバイスを読み込んだりするので, windowsの方が都合がいいので,OS変更はできないんです. でも,OSによる違いが生まれるという知識は今後大いに役立つと思います. ありがとうございました.
- titokani
- ベストアンサー率19% (341/1726)
>1. Sleep関数に含まれる誤差 >2. QueryPerformanceCounter関数に含まれる誤差 QueryPerformanceCounterの値は、CPUのクロックに使われている水晶発振子からきているので、誤差はほぼゼロと考えてよいです。 この場合の原因はSleepの誤差ですね。 理由は他の方もおっしゃっている通りで、Sleepとはそもそもそんなものと考えて使うべきでしょう。 なるべく正確に500msウェイトしたいなら、Sleepで490msとか480msとか待つようにして、最終調整にQueryPerformanceCounterを使うのがいいと思います。 まあ、それでも他のアプリとの競合次第では、大幅にタイミングが遅れる可能性はあります。 そのあたりをどの程度まで許容できるかですね。 絶対に500msを外せないのであれば、OSの選定から考えなければならないでしょう。
お礼
説明が詳しくてわかりやすくて助かります. なるほど,CPUのクロックに用いられているものなら,間違いないでしょうね. 0.1s以下は正直誤差があっても問題はないのですが, 質問されたときに正確に応えたかったもので... 参考になりました.ありがとうございます.
- 正親町(@Ohgimachi)
- ベストアンサー率43% (110/252)
UNIXやWindowsなどのOSでは、1つのハードウエアのタイマーやクロックはOSが独占使用します。そしてアプリには複数存在するように見せかける、仮想的なソフトウェア・タイマーを搭載しています。その仕組みによって複数のプロセスからの要求を実現しているのです。 従って、その時の状況に応じた値になってしまい、必ずしも正確なものを保証している訳ではありません。
お礼
No2さん同様ですが,プロセス管理や優先度を失念していました. お教えいただき,ありがとうございます.
- notnot
- ベストアンサー率47% (4900/10358)
No1の方が書いているように、厳密には無理なわけですが、出来るだけ誤差を減らすためにはそのプロセスのCPUプライオリティを上げればいいです。Win32APIでプロセス起動の時に指定出来たはず。起動してからで良ければタスクマネージャで変更出来ます。 シングルコアで高い優先度にしたプロセスがループすると電源切るしかないので、注意が必要ですが。
お礼
そういえば、プロセスの優先度を高くするっていう方法がありましたね. 失念していました.ありがとうございます.
お礼
Sleep誤差ですか.参考になりました. NASAのリアルタイムOSは気になりますね! 少し調べてみます.ありがとうございます,