What does "DoEvents" do in vb6 ? Why do I get the error message "Out of stack space" ? What does it mean ?
DoEvents() allows other Windows messages to be processed.
The reason you get an out of stack space error is probably because DoEvents() is allowing events to occur that call your code again, which again calls DoEvents(), and so on until the stack space, which tracks the return addresses for all these calls, has run out.
In general, I do not recommend using DoEvents() due to problems like these and the fact that it violates the overall event-driven design of Windows.
DoEvents()
is actually an excellent way to de-prioritize a long running process. And allow the GUI to function. Use them freely but be careful of buttons and other user functions that can trigger something twice. It's pretty much the only way to give the user a cancel button on a long running process. –
Dielectric DoEvents
can cause a lot of problems and it is often used by people who don't understand all the ramifications. –
Salzburg A slightly different way of looking at DoEvents is that it flushes the events in the event queue. If your sub or function triggers an event, that event handler becomes a sub that is in line to run as soon as your sub/function is finished. DoEvents says to run that event handler sub now, instead of waiting till the end of your sub.
While I agree in spirit with Jonathon about not using DoEvents, I would temper his statement by saying I only recommend using it if you know exactly why, and know all of the repercussions of changing the order of the event queue this way. Most often, DoEvents is indicated when you want to update your screen in some way from within the context of a subroutine, before the subroutine is finished executing.
An example of this is when you are using the ProgressBar control. Suppose you are iterating through several thousand records, and want to provide feedback to the user as to how far along you are by updating a progress bar. You might interrupt your loop every hundred records and change the value on the progressbar control. However (unless you do something about it) you won't see the change on the screen until after the progressbar's change event handler runs, and that handler won't run until your sub is done executing. It will just get put in the event queue. The way to force the change event to run immediately, suspending your sub, is to call DoEvents. This will flush all existing events from the queue--in this case your progressbar's change event--and will update the progressbar control on the screen.
Now, "out of stack space" basically means that you've been caught in an endless loop of function calls. The most basic way to cause that is this:
Public sub MySub()
MySub
End Sub
And then call MySub from somewhere. You'll get an out of stack space error. If you look at the Call Stack, you'll see a very long line of calls to MySub.
A well-known real-world example of this would happen in older versions of VB:
Public Sub TextBoxArray_LostFocus(index as Integer)
If TextBoxArray(index) = "" Then
TextBoxArray(index).SetFocus
MsgBox "Please enter a value"
End If
End Sub
This situation assumes two members of a TextBox control array called TextBoxArray. Now, if the user starts with the first one (index 0) and moves to the second one (index 1) then index 0's LostFocus event will fire. However, VB would also internally set the focus to the index 1 box. Then the code would set the focus back to index 0, firing index 1's LostFocus event! You're caught in a loop. They fixed that in VB5 or 6 by waiting to set the focus until the LostFocus event was done executing.
DoEvents
flushes the event queue of the application you are writing in VB6. –
Biltong DoEvents
is it waits for Application.SendKeys()
to finish but it won't wait for e.g. keybd_event()
to finish (stdWindow
uses keybd_event
under the hood). I have no clue how DoEvents is waiting for these events to be processed, as that's something I deamed not easily doable via WinAPI. Just wondered whether you knew how this worked and/or if there is a set of API calls which do the same as DoEvents
does? –
Pringle MsgBox
function. To find it, go to robertrodes.com/work-samples, select Code
, scroll down to Using CBT Hooking to Change the VB MsgBox Function and expand it. HTH –
Biltong DoEvents
won't wait for keybd_event()
, given keybd_event
does create WM_KEYUP
and WM_KEYDOWN
events..., Though it could be that these just aren't tracked by the VBX Application perhaps... Or weren't added to the key event queue in the first place? –
Pringle WH_KEYBOARD
instead of WH_CBT
with SetWindowsHookEx
for what you want to do. The CBT one, when I look it up, is more for window manipulation. Here's the Microsoft doc on the different hooks available (I think you've already found this, but just in case). –
Biltong I would clarify Johnathon's answer in that it pumps that VB message loop and allows the VB Runtime to process windows messages, which is the opposite of Sleep which allows for Windows to process its events (not necessary in the world of Multicore CPUs and true multitasking OS's but when VB6 was written Windows 9x was the dominant OS and a hard loop that only had DoEvents in it would spike the CPU usage to 100%). So seeing things like
While fDoneFile = False
DoEvents
Sleep 55
Wend
was a common pattern throughout the VB6 world.
As stated else where, DoEvents allows other events in your application to fire. Here's an example of how you can use DoEvents without the "Out of stack space" issue. This makes sure you don't run through the code multiple times by using a Boolean to indicate the code is running.
Sub Example()
'Create static variable to indicate the sub is running.
Static isRunning As Boolean
'Exit the sub if isRunning
If isRunning Then Exit Sub
'Indicate sub is running
isRunning = True
'Sub does stuff
DoEvents
'Ends up calling sub again
Example 'Added just to prove via testing.
'Indicate sub is no longer runningrunning
isRunning = False
End Sub
© 2022 - 2024 — McMap. All rights reserved.