

ボールがあったとしてWM_TIMERによって
一定の速度でy方向に動いていたとします。
ある条件(例えば5回に1度とか)で
y方向に行くのを一旦やめて
x方向に一定の間隔で5マス移動するとします。
while(aaa() > 0){
x++;
Sleep(100);
InvalidateRect(hWnd, NULL, TRUE);
}
このようなプログラムです。
aaa()は5回呼ばれると0を返します。
これでx==0から始まったとすれば
x==1,x==2,x==3,・・・・とボールが移動する様子が表示されると
思っていたのですが、
実際には5回Sleep(100)したくらいの時間が経ってから
一気にx==5の位置まで移動して表示されました。
デバッグして追ってみたら
while文は5回ループして
InvalidateRect(hWnd, NULL, TRUE)もちゃんと
5回行われているようでした。
どういうことなんでしょうか?
No.5ベストアンサー
- 回答日時:
>UpdateWindow()がキューに入れないで直接プロシージャにWM_PAINTを送信するっていうのは
>わかったのですがなぜUpdateWindow()だけだといけないのでしょうか?
「無効領域があった場合にWM_PAINTを送信」だからでしょう。
http://msdn.microsoft.com/ja-jp/library/cc428780 …
に
>指定されたウィンドウの更新リージョンが空ではない場合
とありますし。
無駄にWM_PAINTは抑える方向に考えられている。ということかと。
>WM_TIMER内でwhileを回していました。
メッセージを回すのが止まってしまうので出来れば避けた方がよいかと……。
スレッドにする。という方法があるかと思われます。
# 排他制御とか必要になりますが…。
No.3
- 回答日時:
Sleep(100);は確かにWindowsに制御は返しますがWM_TIMERメッセージの処理から抜けているわけではないので、WindowsはWM_Paint処理を呼ぶことは出来ません。
すなわち描画は更新されません。WM_TIMERメッセージの処理の中でSleepするのではなく、グローバル変数に現在の状態を記憶し毎回WM_TIMERメッセージの処理を抜け、WM_TIMERメッセージの処理に入った時にグローバル変数の値によって処理を分岐するような形にする必要があります。いわゆる状態遷移表によるプログラミングになります。Windowsのプログラミングは面倒です。
No.2
- 回答日時:
InvalidateRectは、ウィンドウの無効領域を更新します。
その結果、ウィンドウに再描画が必要になった場合は、WM_PAINTメッセージが発行されます。ただし、すでにメッセージキューにWM_PAINTがある場合は、新たにWM_PAINTは発生しません。ウィンドウの描画処理は、WM_PAINTメッセージを処理することで行われるので、whileループがメインスレッドにある場合、他のメッセージ処理を行う機会がないため、whileループが終了するまで画面が更新されません。
Sleepで時間稼ぎをすると、そのスレッドは停止状態になり、何も処理できなくなるので、少なくともメインスレッドでは使うべきではないと思います。SetTimerでは問題があるのであれば、少なくともSleepする前にUpdateWindowなどメッセージ処理以外の機会でウィンドウの更新をするべきでしょう。
No.1
- 回答日時:
複数のWM_PAINTは纏められる場合がありますけど…その辺はどうなんでしょう?
InvalidateRect(hWnd, NULL, TRUE)で無効領域(再描画が必要な領域)が設定されて、
メッセージキューに積まれます。
メッセージキューにWM_PAINTが残っている状態でInvalidateRect(hWnd, NULL, TRUE)で
再度無効領域を設定すると、WM_PAINTはどっちか1つだけ…ということになるかと。
# いくつか進んでいるキューの途中のWM_PAINTが残って最新のInvalidateRect()分がキューに積まれないか、
# 途中のWM_PAINTがキューから削除されて、改めてキューの最後にWM_PAINTが入るか……。
InvalidateRect()の後にUpdateWindow()入れたらどうなります?
念のため確認ですが…そのwhileループってなんらかのメッセージ処理中(ボタンクリックとかWM_TIMERメッセージとか)で回していたりしないですよね?
WM_TIMERのハンドラ内で回していると、WM_PAINTはまず間違いなく1つに纏められますよ。
# タイミングによっては「応答なし」がウィンドウタイトルに付くかも知れません。
この回答への補足
回答ありがとうございます。
WM_TIMER内でwhileを回していました。
InvalidateRect()はWM_PAINTメッセージをキューに入れるので
他のメッセージの最中に5回呼び出すっていうのは
WM_PAINTの内容を実行させずただキューに5個入れるようとしちゃってたんですね。
実際は1つにまとめられるようですが。理解できました。
>>InvalidateRect()の後にUpdateWindow()入れたらどうなります?
やってみたら5回更新されていました。
ただInvalidateRect()を消してみたら先ほどと同じ状態になってしまいました。
UpdateWindow()がキューに入れないで直接プロシージャにWM_PAINTを送信するっていうのは
わかったのですがなぜUpdateWindow()だけだといけないのでしょうか?
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
このQ&Aを見た人はこんなQ&Aも見ています
関連するカテゴリからQ&Aを探す
おすすめ情報
このQ&Aを見た人がよく見るQ&A
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
ACCESS側からEXCELの書式を設定...
-
VCでウエイトをミリ秒でかけ...
-
VBA、UserFormを前面に出力して...
-
【C#】 あるイベントから別イ...
-
VBSで応答不要のメッセージボッ...
-
VBA メッセージボックスを自動...
-
Application.ScreenUpdating=Fa...
-
Stack around the variable 'xx...
-
VBSの処理中一旦処理を止めて再...
-
「キャンセル」ボタン付きの処...
-
ASP.NETでのメッセージ画面を出...
-
マウスのクリックを無視したい
-
C#で処理中画面を表示したい。
-
thunderbirdのメッセージフィル...
-
ファンクションキーのキャンセ...
-
Macターミナルで実行中のプログ...
-
C# シリアル通信でデータ受信...
-
バックグラウンドのプロセスの...
-
C言語で、メモリを解放しないで...
-
TCP/IP通信時のサーバーからの受信
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
VBSの処理中一旦処理を止めて再...
-
VBA kernel32 の意味
-
メッセージボックスのボタン名変更
-
ACCESS側からEXCELの書式を設定...
-
VBSで応答不要のメッセージボッ...
-
VBA、UserFormを前面に出力して...
-
【C#】 あるイベントから別イ...
-
Excel VBA で処理中断(DoEvents...
-
PostMessageの連続送信
-
ボタンが押された事を検知する...
-
Application.ScreenUpdating=Fa...
-
エクセルVBAでクリップボード内...
-
VB Loopを一時停止し再開あるい...
-
起動後直に実行するコードはど...
-
DirectX環境下での方向キー同時...
-
Excel VBA 実行中に一瞬フリー...
-
OSシャットダウン時の常駐アプ...
-
マウスのクリックを無視したい
-
エクセルVBAで
-
メッセージボックスの表示につ...
おすすめ情報