Rubrik: Variablen/Strings · Sonstiges | VB-Versionen: VB6 | 14.10.11 |
DoEvents gezielt und nur wenn notwendig einsetzen Hier wird gezeigt, wie man auf DoEvents bei zeitkritischen bzw. sehr langen Schleifen / Vorgängen gezielt einsetzt, ohne unnötig Zeit zu vergeuten. | ||
Autor: Dieter Otter | Bewertung: | Views: 34.573 |
www.tools4vb.de | System: Win9x, WinNT, WinXP, Win7, Win8, Win10, Win11 | Beispielprojekt auf CD |
Sie kennen das sicherlich: Während eines längeren Vorgangs soll dem Benutzer die Möglichkeit gegeben werden, den Vorgang bspw. durch Klick auf einen Button abbrechen zu können.
Während der Vorgang läuft reagiert das Programm jedoch nicht auf Benutzer-Aktionen. Überall ist zu lesen, dass hier DoEvents das Allheilmittel ist!
In gewisser Weise stimmt das auch! Allerdings sollte mit die DoEvents-Anweisung "sparsam" eingesetzt werden, da diese den gesamten Vorgang enorm verlangsamt.
Beispiel: Es werden sehr viele Datensätze innerhalb einer Schleife verarbeitet. Der Vorgang soll jederzeit vom Benutzer angehalten / abgebrochen werden können. Wir setzen also ein DoEvents in die Schleife, damit wir auf einen Button-Klick (Abbruch) reagieren können.
bAbort = False Do ' Aktion ... ' ... DoEvents ' Damit man auf einen Button-Klick oder ähnliches reagieren kann If bAbort Then Exit Do Loop Until ...
Im Click-Event des Abbrechen-Buttons steht:
Private Sub btnAbort_Click() If MsgBox("Wollen Sie den Vorgang wirklich abbrechen?", vbQuestion Or vbYesNo) = vbYes Then ' Abbrechen bAbort = True End If End Sub
Mit obigen Code wird bei jeder einzelnen Aktion ein DoEvents aufgerufen. Dies verlangsamt den Prozess bei sehr vielen Schleifendurchläufen gewaltig!
Besser ist es, wenn man DoEvents bspw. nur alle x - Schleifendurchläufe aufruft:
Dim nCount As Long bAbort = False nCount = 0 Do ' Aktion ... ' ... nCount = nCount + 1 ' DoEvents nur alle 10 Schleifendurchläufe aufrufen If nCount Mod 10 = 0 Then DoEvents If bAbort Then Exit Do Loop Until ...
Das Aufrufen von DoEvents alle x-Schleifendruchläufe verbessert die Performance schon enorm.
Noch besser wäre es aber, wenn man DoEvents nur dann aufruft, wenn der User auch tatsächlich eine Taste oder einen Mausklick getätigt hat - oder?
Genau für diesen Fall gibt es die GetInputState API-Funktion. Diese Funktion liefert einen Wert ungleich Null zurück, wenn ein Tastatur- oder Mausklick-Ereignis erfolgte.
Option Explicit ' benötigte API-Deklarationen Public Declare Function GetInputState Lib "user32" () As Long
Wir ändern den Code der Schleife also wie folgt ab, um die beste Performance zu erzielen:
Dim nCount As Long bAbort = False Do ' Aktion ... ' ... ' Falls Tastatur-/Mausklick-Ereignis vorliegt If GetInputState() <> 0 Then DoEvents If bAbort Then Exit Do End If Loop Until ...