- •Лекция 13
- •XTimers.Vbp 5
- •Обзор задачи Coffee Sample Application
- •XTimers.Vbp
- •XTimerSupport.Bas
- •Private Type xtimerinfo ' Hungarian xti
- •Public Sub TimerProc(ByVal hWnd As Long, ByVal uMsg As Long, ByVal idEvent As Long, ByVal lngSysTime As Long)
- •If maxti(intCt).BlnReentered Then Exit Sub
- •Public Sub EndTimer(ByVal xt As xTimer)
- •Public Sub Scrub()
- •Public Property Get Interval() As Long
- •Public Property Let Interval(ByVal NewInterval As Long)
- •Public Sub RaiseTick()
- •Private Sub Class_Terminate()
- •MtCoffee.Vbp modMt.Bas Option Explicit
- •Public glngGlobalData As Long
- •Coffee.Bas
- •Option Explicit
- •Private Declare Function timeGetTime Lib "winmm.Dll" () As Long
- •Private mlngIterations As Long
- •Public Property Get NumberOnThread() As Long
- •Public Sub StartLongTask(ByVal Iterations As Long)
- •Public Function GetCoffeeOnSameThread() As Coffee
- •Public Function GetCoffeeOnNewThread() As Coffee
- •If blnCancel Then
- •Coffee.Cls Option Explicit Private Declare Function timeGetTime Lib "winmm.Dll" () As Long
- •Private mlngIterations As Long
- •Public Property Get NumberOnThread() As Long
- •Public Sub StartLongTask(ByVal Iterations As Long)
- •Public Function GetCoffeeOnSameThread() As Coffee
- •Public Function GetCoffeeOnNewThread() As Coffee
- •If blnCancel Then
- •Public Property Get CoffeeMonitor() As CoffeeMonitor
- •If gCoffeeMonitor Is Nothing Then
- •CoffeeMonitor.Cls Option Explicit
- •Private WithEvents mwXTimer As xTimer
- •Event CoffeeReady() Private Sub Class_Initialize()
- •Private Sub Class_Terminate()
- •Private Sub mwXTimer_Tick()
- •Connector2.Cls Option Explicit
- •Public Property Get CoffeeMonitor2() As CoffeeMonitor2
- •CoffeeMonitor2 Option Explicit
- •Private maicnClients() As iCoffeeNotify
- •Public Sub TellMeReady(ByVal icn As iCoffeeNotify)
- •Public Sub CeaseCallBacks(ByVal icn As iCoffeeNotify)
- •Private Sub mwXTimer_Tick()
- •ICoffeeNotify.Cls
- •Public ThreadId As Long
- •Private Sub mwCoffee_Progress(ByVal PercentDone As Single, Cancel As Boolean)
- •Private Sub cmdCallBacks_Click()
- •Private Sub cmdMt_Click()
- •Private Sub Form_Unload(Cancel As Integer)
- •Private Sub mwcmnEvents_CoffeeReady()
- •FrmThread.Frm
- •Private Sub cmdCancel_Click()
- •Private Sub cmdShortOnly_Click()
- •Private Sub cmdShortLong_Click()
- •Private Sub cmdNLong_Click()
- •Private Sub cmdXThread_Click()
- •Private Sub cmdIDs_Click()
- •Private Sub tmrShort_Timer()
Public Sub StartLongTask(ByVal Iterations As Long)
' This is a short circuit for testing call overhead. See CallAnotherCoffee.
If Iterations = 0 Then Exit Sub
' Store the size of the dummy task.
mlngIterations = Iterations
' Give the timer a short interval, and set it running just before returning.
mwXTimer.Interval = 55
mwXTimer.Enabled = True
End Sub
' GetCoffeeOnSameThread creates a new Coffee object on the same thread, simulating the effects of thread pooling. This can only be done internally, as explained in "How Object Creation Works in Visual Basic" in Books Online.
'
Public Function GetCoffeeOnSameThread() As Coffee
' All objects created using New will be on the creator's thread, even a new Coffee object.
Set GetCoffeeOnSameThread = New Coffee
End Function
' GetCoffeeOnNewThread creates a new Coffee object on a new thread, by calling CreateObject to create the new Coffee object. The difference between this and the internal creation done by GetCoffeeOnSameThread is explained in "How Object Creation Works in Visual Basic" in Books Online.
'
' Note that this technique could be used to create objects on different threads that could communicate with each other, without the client having to pass one object a reference to the other (as CoffeeWatch does). If you experiment with this, remember that the overhead of marshaling calls between threads is almost as great as the overhead of marshaling calls across processes.
'
Public Function GetCoffeeOnNewThread() As Coffee
' Create as if by external client.
Set GetCoffeeOnNewThread = CreateObject("MTCoffee.Coffee")
End Function
' CallAnotherCoffee gives a rough measure of cross-thread call overhead. Pass it a Coffee object on another thread, or on the same thread, and compare the results; the method makes dummy calls to StartLongTask, so that it's essentially measuring only the call overhead.
'
Public Function CallAnotherCoffee(ByVal cfe As Coffee) As Double
Const TRIES = 10000
Dim timeStart As Long
Dim timeEnd As Long
Dim lngTries As Long
timeStart = timeGetTime
For lngTries = 1 To TRIES
cfe.StartLongTask 0
Next
timeEnd = timeGetTime
'
' Return seconds (ss.mmm) per call. (This will give an incorrect result if you happen to run CallAnotherCoffee just as the system timer is rolling over to zero.)
CallAnotherCoffee = ((CDbl(timeEnd) - timeStart) / 1000#) / TRIES
End Function
Private Sub Class_Initialize()
' Increment the global count (that is, for this thread) of Coffee objects.
glngGlobalData = glngGlobalData + 1
'
' Create a timer object.
Set mwXTimer = New XTimer
End Sub
Private Sub Class_Terminate()
' Decrement the global count (that is, for this thread) of Coffee objects.
glngGlobalData = glngGlobalData - 1
'
' Free the timer object.
Set mwXTimer = Nothing
End Sub
Private Sub mwXTimer_Tick()
' First thing, turn off the timer.
mwXTimer.Enabled = False
Call LongTask
End Sub
' The dummy task.
Private Sub LongTask()
Dim dblDummy As Double
Dim lngCt As Long
Dim sngNextMark As Single
Dim blnCancel As Boolean
' For small transactions, don't bother to call back while running.
If mlngIterations < 100000 Then
sngNextMark = 1!
Else
sngNextMark = 0.1!
End If
' This is just a time-waster.
For lngCt = 1 To mlngIterations
' If this were a real application, a unit of work would be done here. You may find it interesting to replace this processor-intensive activity with one that waits on the system a lot, such as calls to a database on another machine, or reading a very large file. Throughput on a single-processor workstation is far greater when most threads are blocked, waiting for file input or the result of a database call.
dblDummy = 3033.14159 * 2081.14159 * 1138.14159
If CDbl(lngCt) / mlngIterations > sngNextMark Then
RaiseEvent Progress(sngNextMark, blnCancel)