• <ruby id="5koa6"></ruby>
    <ruby id="5koa6"><option id="5koa6"><thead id="5koa6"></thead></option></ruby>

    <progress id="5koa6"></progress>

  • <strong id="5koa6"></strong>
    • 軟件測試技術
    • 軟件測試博客
    • 軟件測試視頻
    • 開源軟件測試技術
    • 軟件測試論壇
    • 軟件測試沙龍
    • 軟件測試資料下載
    • 軟件測試雜志
    • 軟件測試人才招聘
      暫時沒有公告

    字號: | 推薦給好友 上一篇 | 下一篇

    Doevents函數詳解

    發布: 2007-5-25 09:19 | 作者: 未知 | 來源: programfan.com | 查看: 56次 | 進入軟件測試論壇討論

    領測軟件測試網
    Doevents函數是一個很好用的函數,但很多人對它的用法不清楚或有誤解。由于我在網上查到一篇關于此函數的用法,并添加了一些內容,不敢獨享,特此獻出。
        其中有一個“控時循環和變速齒輪”的內容,有點意思,感興趣的可看一看。

        DoEvents函數的功能是:轉讓控制權,以便讓操作系統處理其它的事件。
        問:為什么要用doevents?
         A.在需要用某一循環處理相當耗時或者很快速的代碼時,就需要用到它,以便用戶能在起處理過程中能做其他事情,即程序能被控制,而不是無響應狀態
         B.vb6.0中多線程vb代碼極度不穩定,而且無法調試,所以vb中的多線程用的很少(注:是指vb的代碼在多線程中運行時不穩定)
         C.timer控件可以起到后臺運行作用,但其是通過事件控制,一是不穩定,二是速度太慢,如果想用其處理高速又耗系統的代碼更本不能達到預期的效果

    下面將其某些用法和難點簡介如下:
         
        (注: 1.'** 后面的代碼表示如果在該處用了這個語句。2.例子中會用到API函數。3.以下例子都經vb6.0測試成功)
         
    一. 基本用法:
        1.窗體啟動時如果要處理的事務太多或者用sleep函數暫停,造成其很久都不能出現時怎么辦?
        例如代碼:
        Public Declare Sub Sleep Lib "kernel32" (ByVal dwMilliseconds As Long)'此句寫入模塊
        Private Sub Form_Load()
        Show
        '**DoEvents  句3
        Sleep 5000
        End Sub
        通常容易想到在sleep前加個show,但還是不能達到預想的效果,窗體雖然出來了,但好象只達到了一半,如果加上第3句,將看到效果大不相同 。

        2.如果有個很耗時的循環導致程序不響應,怎么辦?
        例如:
        Dim L As Long
        For L = 1 To 1000000
        '** DoEvents
        Next L
        如果無'**,在循環過程中程序無法處理事件,對于用戶來說是不響應,無法控制的

        3.想在循環中看到處理過程?
        同樣:
        Dim L As Long
        For L = 1 To 10000
        '** DoEvents
        Text1.Text=Cstr(l)
        Next L
        無'** 時將無法看到text1中的變化,而只在循環結束時看到最后結果

        4.怎樣中止循環?
        如果有:
        Private Sub Command1_Click()
        Dim L As Long
        Do
         L = L + 1
         Debug.Print L  '在立即窗口中顯示
         DoEvents
        Loop
        End Sub
        會發現當關閉窗口后,debug中的數據仍然在變化,說明并沒結束
        需要如下:
        Dim IsExit As Boolean
        Private Sub Command1_Click()
        Dim L As Long
         IsExit = False
         Do While DoEvents
         If IsExit = True Then Exit Do
         L = L + 1
         Loop
        End Sub
         
        Private Sub Command2_Click()'或者在form_unload模塊中等等
         IsExit = True
        End Sub
        其中 isexit是全局變量
        <>有些人喜歡用end語句來結束程序,小程序固然可以,但當太大,或者調用了某些特殊的api函數后可能導致預想不到的錯誤,如果裝載了許多東西在程序結束時不處理將卸載很慢,而且這種做法也極不符合正規軟件的要求...總之end語句毛病很多,此不詳談,建議少使用甚至不使用

    二. 其基本用法大概就這些,現在解析其中的一些[難點]
         
        1.為什么還是不能結束?
        代碼如下:
        Dim IsExit As Boolean
        Private Sub Command1_Click()
        Dim L As Long
         IsExit = False
         Do
         If IsExit = True Then Exit Do '句0
         DoEvents '** 句1
         Text1.Text = CStr(L) '** 句2
         L = L + 1
         Loop
        End Sub
         
        Private Sub Form_Load()
        Static N As Long
        N = N + 1
        MsgBox N
        End Sub
         
        Private Sub Form_Unload(Cancel As Integer)
         IsExit = True
        End Sub
         
        運行結果:啟動時msg顯示1,點擊command1,text1在變化,此時再點form右上角的小差(關閉窗體),發現vb運行控制上的按扭并沒變化,說明程序還在運行.如果編譯成程序后運行,按下ctrl+del+alt也可發現它還沒結束.
        通過讀代碼,并沒發現錯誤,怎么回事?
        關鍵在于"句2"訪問了控件的屬性 :
        代碼運行路徑:當在doevents 時,程序釋放控制權,可以接收事件消息,form-unload事件只能從此處產生,假設此時關閉form ,unload事件發生,即doevents后就運行unload代碼,得到isexit=t,并且form卸載,代碼返回到doevents 之后,運行句2.注意現在form 已經卸載了,text1從哪里來呢? 于是form重新裝載,代碼跳到form_load模塊運行,所以在關閉窗體后可以看到msg 顯示2,此模塊運行完后再繼續句2后面的代碼,當下次循環遇到 句0時退出循環
        另:既然退出了循環,怎么還不能結束?
        vb程序規定(其實其他的windows語言一樣):窗體卸載時并不是立即卸載其模塊代碼,而只先卸載窗體中的控件和一些屬性值,程序中最后一個窗體卸載時才完全卸載.
        在這個單窗體程序中,form卸載時因為循環的控制無法卸載代碼,失去了卸載代碼的機會,導致再也不能卸載(因為沒卸載代碼,所以運行的句2是并不會出錯) 。
        另:既然再次運行了form_load代碼,怎么看不見窗體?
        因為程序啟動時窗體的到顯示的消息,而只運行此模塊并沒有(如果在msgbox n語句前加上show,就可以看到它了)
        如何解決?
        通過以上分析,應該很簡單,把句1 和句2調換一下就可以了,關鍵:
        <仔細分析代碼是如何運行的,避免在form已經卸載了情況下訪問控件>

        2.用了doevents速度太慢了怎么辦?
        doevents的代價是速度變慢,但要程序響應又不得不用,其實doevents語句允許任何應用程序執行相關事件,而不僅僅是你自己的程序,所以變得很慢.
        可以讓它響應本程序事件動作,需要用到api函數GetInputState,它的聲明語句為:
        Public Declare Sub Sleep Lib "kernel32" (ByVal dwMilliseconds As Long)
        Public Declare Function GetInputState Lib "user32" () As Long
        例如用: If GetInputState() Then DoEvents '來代替doevents可使循環運行更快

        3.既要同時響應事件又要控件不變化,怎么辦?
        例如在一個長的循環中向listview控件中添加記錄,無doevents時程序無響應,但有它時控件又閃的厲害
        解決辦法:
        a.不一定每次循環都doevents,可以在適當時間時才用,至少沒那么閃
        b.應用api函數 ValidateRect 功能是使指定的矩型區域生效,通知Windows不對指定的區域進行重畫 另:InvalidateRect 功能相反,同時需要用到函數 GetClientRect 取得指定對象的矩形區域 應用*rect函數指定listview的矩形區不重畫,即可避免閃爍(但還是要注意恢復重畫,否則看不見了真實效果)

        4.控時循環和變速齒輪
        請看下面的代碼:
        Option Explicit
        Private Declare Function timeGetTime Lib "winmm.dll" () As Long
        Dim IsExit As Boolean
        Private Sub Command1_Click()
        Dim L As Long
        Dim Kt As Long
        IsExit = False
        Do
         Kt = timeGetTime()
         'do something
         L = L + 1
         Text1.Text = L
         'DoEvents '句 1
         While timeGetTime - Kt < 50 '句 2
         'While Abs(timeGetTime - Kt) < 50 '句 3
         'While Abs(timeGetTime - Kt) And (Not IsExit) < 50 '句 4
         
         DoEvents '句 5
         Wend
         'DoEvents '句 6
         If IsExit Then Exit Do
        Loop
        End Sub
         
        Private Sub Form_Unload(Cancel As Integer)
        IsExit = True
        End Sub
        其中可用的代碼(除去加"'" 號的代碼)就是通常的控時循環代碼
        運行代碼并不會出現錯誤,但在循環過程,請開啟變速齒輪看看
        當關閉齒輪時,將發現text1.text停止了,別慌,等一段時間它又會繼續(這要看你設定的時間,這里是50毫秒,如果設定的太長text1.text將半天都沒變化,這是怎么回事?
         變速齒輪在啟動時將hook.dll映射到你的程序地址運行,更改了timegettime()函數獲取的時間
        如果在句2和句3間插入debug.print timegettime,timegettime-kt 將發現,在關閉齒輪的瞬間后者變成了負值,timegettime變小了,所以才造成需要等很久
        如果是編寫游戲,而用戶開了齒輪,那可就慘了
         
        解決方案:
        a.用句3代替句2,這個方法最簡便,雖然不符實,但不會出問題,建議使用
        b.不要句5,換用句6(這樣就能達到效果嗎?) 因為齒輪還是從doevents語句運行時才能插的進來,所以只要kt=timegettime 和 timegettime之間沒有doevents就不會出錯
        ab.兩種方法都有些小問題,但無大礙,有興趣者請自己分析

        5.程序怎么"死了"?
        這只是一些人編寫時沒注意到的小問題,提醒一下:
        同樣用上面的代碼,如果設定的時間太短,以至在代碼運行到句2時已經超時了,句5將不能運行了,當然程序就死了哦,以防萬一,加上句1,所以此時也只能用a方案來解決齒輪的問題了
        有必要用句4代替句3 嗎? 除非你設定的時間太長,人家想關閉你的程序要等上好半天。

     在MSND上的內容:在使用全局數據時避免 DoEvents
    當一個函數已通過 DoEvents 放棄控制時,可相當安全地再次調用函數。例如,下一過程將檢測質數并用 DoEvents 語句周期地啟動其它應用程序處理事件:

    Function PrimeStatus (TestVal As Long) As Integer
       Dim Lim As Integer
       PrimeStatus = True
       Lim = Sqr(TestVal)
       For I = 2 To Lim
          If TestVal Mod I = 0 Then
             PrimeStatus = False
             Exit For
          End If
          If I Mod 200 = 0 Then DoEvents
       Next I
    End Function

    該代碼中每重復 200 次就調用一次 DoEvents 語句。這樣一來,當該環境的其余部分對事件作出響應時,只要有必要,PrimeStatus 過程就可繼續計算。

    考慮在調用 DoEvents 期間發生的事情。在其它窗體和應用程序處理事件時將暫停執行應用程序代碼。這些事件之一有可能是一個按鈕單擊操作,它將再次啟動 PrimeStatus 過程。

    這將導致重新進入 PrimeStatus 過程的,但是,因為在函數每次出現時,堆棧都為其參數和局部變量分配了空間,所以重入不會引發沖突。當然,如果過多調用 PrimeStatus,則可能出現“溢出堆?臻g”錯誤。

    如果 PrimeStatus 使用或改變模塊級變量或全局數據,情況就會完全不同。此時,在 DoEvents 能夠返回之前執行 PrimeStatus 的另一個實例,這將導致模塊數據或全局數據的值完全不同于它們在調用 DoEvents 之前的值。于是,PrimeStatus 的結果將會難以預料。

    延伸閱讀

    文章來源于領測軟件測試網 http://www.kjueaiud.com/


    關于領測軟件測試網 | 領測軟件測試網合作伙伴 | 廣告服務 | 投稿指南 | 聯系我們 | 網站地圖 | 友情鏈接
    版權所有(C) 2003-2010 TestAge(領測軟件測試網)|領測國際科技(北京)有限公司|軟件測試工程師培訓網 All Rights Reserved
    北京市海淀區中關村南大街9號北京理工科技大廈1402室 京ICP備10010545號-5
    技術支持和業務聯系:info@testage.com.cn 電話:010-51297073

    軟件測試 | 領測國際ISTQBISTQB官網TMMiTMMi認證國際軟件測試工程師認證領測軟件測試網

    老湿亚洲永久精品ww47香蕉图片_日韩欧美中文字幕北美法律_国产AV永久无码天堂影院_久久婷婷综合色丁香五月

  • <ruby id="5koa6"></ruby>
    <ruby id="5koa6"><option id="5koa6"><thead id="5koa6"></thead></option></ruby>

    <progress id="5koa6"></progress>

  • <strong id="5koa6"></strong>