日韩欧美人妻无码精品白浆,夜夜嗨AV免费入口,国产欧美官网在线看,高校回应聋哑女生因长相完美被质疑

LOGO OA教程 ERP教程 模切知識(shí)交流 PMS教程 CRM教程 開發(fā)文檔 其他文檔  
 
網(wǎng)站管理員

VB-程序運(yùn)行沒有響應(yīng)的解決辦法-DoEvnts

admin
2014年3月13日 8:11 本文熱度 7337

在VB編程的時(shí)候,很多人肯定碰到過開啟程序以后程序界面會(huì)“沒反應(yīng)”,“沒有響應(yīng)”,但是我們代碼要求執(zhí)行的任務(wù)還在正常的進(jìn)行,任務(wù)執(zhí)行完畢,程序界面又恢復(fù)正常,尤其是使用循環(huán)的時(shí)候很容易出現(xiàn)程序沒響應(yīng)的現(xiàn)象,這類問題怎么解決呢?用DoEvents就可以解決。舉例:


Private Sub DelayTime(num As Long)
Dim star, ends As Long
star = timeGetTime
ends = star
Do While ends - star < num
ends = timeGetTime
DoEvents
Loop
End Sub


DoEvents 是: 轉(zhuǎn)讓控制權(quán),以便讓操作系統(tǒng)處理其它的事件。最簡(jiǎn)單的理解,比如你要在某個(gè)耗時(shí)很多的過程中(最常見的是循環(huán)),還要響應(yīng)某個(gè)操作-比如控制進(jìn)度條的顯示,那就需要加入DoEvents
但是,當(dāng)看到這句話時(shí),腦子里面的概念很模糊,只有一個(gè)淺薄的認(rèn)識(shí)。打個(gè)比方:如果現(xiàn)在有一個(gè)從1 到100000000的循環(huán),有一點(diǎn)編程常識(shí)的人肯定知道在這個(gè)耗時(shí)較長(zhǎng)的循環(huán)過程中,基本上是不能進(jìn)行別的操作(當(dāng)然不排除一些特別的情況),但是因?yàn)槟承┨貏e的要求,你要在循環(huán)過程中,進(jìn)行其他操作,這時(shí)DoEvents 就派上用場(chǎng)了。
現(xiàn)在,大家應(yīng)該明白了DoEvents 的主要功能了吧。下面舉一些實(shí)例(基本上都是轉(zhuǎn)載的),當(dāng)然建議自己能親自去試驗(yàn),理解會(huì)深一些。

(注: '** 后面的代碼表示如果在該處用了這個(gè)語句 以下代碼中用到了一些api函數(shù),請(qǐng)用vb附帶的api瀏覽器查閱) (轉(zhuǎn)載)

一. 基本用法:
1.窗體啟動(dòng)時(shí)如果要處理的事務(wù)太多或者用sleep函數(shù)暫停,造成其很久都不能出現(xiàn)時(shí)怎么辦?
例如代碼:
Private Sub Form_Load()
Show
'DoEvents
Sleep 5000
End Sub
通常容易想到在sleep前加個(gè)show,但還是不能達(dá)到預(yù)想的效果,窗體雖然出來了,但好象只達(dá)到了一半,如果加上第3句,將看到效果大不相同

2.如果有個(gè)很耗時(shí)的循環(huán)導(dǎo)致程序不響應(yīng),怎么辦?
例如:
Dim L As Long
For L = 1 To 1000000
'DoEvents
Next L
如果無'DoEvents,在循環(huán)過程中程序無法處理事件,對(duì)于用戶來說是不響應(yīng),無法控制的


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


4.怎樣中止循環(huán)?
如果有:
Private Sub Command3_Click()
Dim L As Long
Do
L = L + 1
Debug.Print L
DoEvents
Loop
End Sub
會(huì)發(fā)現(xiàn)當(dāng)關(guān)閉窗口后,debug中的數(shù)據(jù)仍然在變化,說明并沒結(jié)束
需要如下:
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語句來結(jié)束程序,小程序固然可以,但當(dāng)太大,或者調(diào)用了某些特殊的api函數(shù)后可能導(dǎo)致預(yù)想不到的錯(cuò)誤。如果裝載了許多東西在程序結(jié)束時(shí)不處理將卸載很慢,而且這種做法也極不符合正規(guī)軟件的要求。總之end語句毛病很多,此不詳談,建議少使用甚至不使用。

二. 其基本用法大概就這些,現(xiàn)在解析其中的一些[難點(diǎn)]

1.為什么還是不能結(jié)束?
代碼如下:
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

運(yùn)行結(jié)果:?jiǎn)?dòng)時(shí)msg顯示1,點(diǎn)擊command1,text1在變化。此時(shí)再點(diǎn)form右上角的小差(關(guān)閉窗體),發(fā)現(xiàn)vb運(yùn)行控制上的按扭并沒變化,說明程序還在運(yùn)行。如果編譯成程序后運(yùn)行,按下ctrl+del+alt也可發(fā)現(xiàn)它還沒結(jié)束。通過讀代碼,并沒發(fā)現(xiàn)錯(cuò)誤,怎么回事?

關(guān)鍵在于 句2 訪問了控件的屬性
代 碼運(yùn)行路徑:當(dāng)在DoEvents時(shí),程序釋放控制權(quán),可以接收事件消息,form-unload事件只能從此處產(chǎn)生,假設(shè)此時(shí)關(guān)閉form,unload事件發(fā)生,即DoEvents后就運(yùn)行unload代碼,得到isexit=t,并且form卸載,代碼返回到DoEvents之后,運(yùn)行 句2,注意現(xiàn)在form 已經(jīng)卸載了,text1從哪里來呢?


于是form重新裝載,代碼跳到form_load模塊運(yùn)行,所以在關(guān)閉窗體后可以看到msg 顯示2,此模塊運(yùn)行完后再繼續(xù)句2后面的代碼,當(dāng)下次循環(huán)遇到 句0時(shí)退出循環(huán)


另:既然退出了循環(huán),怎么還不能結(jié)束?
vb程序規(guī)定(其實(shí)其他的windows語言一樣):窗體卸載時(shí)并不是立即卸載其模塊代碼,而只先卸載窗體中的控件和一些屬性值,程序中最后一個(gè)窗體卸載時(shí)才完全卸載。
在這個(gè)單窗體程序中,form卸載時(shí)因?yàn)檠h(huán)的控制無法卸載代碼,失去了卸載代碼的機(jī)會(huì),導(dǎo)致再也不能卸載(因?yàn)闆]卸載代碼,所以運(yùn)行的 句2 是并不會(huì)出錯(cuò))


另:既然再次運(yùn)行了form_load代碼,怎么看不見窗體?
因?yàn)槌绦騿?dòng)時(shí)窗體的到顯示的消息,而只運(yùn)行此模塊并沒有(如果在msgbox n語句前加上show,就可以看到它了)
如何解決?
通過以上分析,應(yīng)該很簡(jiǎn)單,把句1 和句2調(diào)換一下就可以了,關(guān)鍵:
<仔細(xì)分析代碼是如何運(yùn)行的,避免在form已經(jīng)卸載了情況下訪問控件>

2.用了DoEvents速度太慢了怎么辦?
DoEvents的代價(jià)是速度變慢,但要程序響應(yīng)又不得不用
其實(shí)DoEvents語句允許任何應(yīng)用程序執(zhí)行相關(guān)事件,而不僅僅是你自己的程序,所以變得很慢。
可以讓它響應(yīng)本程序事件動(dòng)作,需要用到api函數(shù)GetInputState
例如用: If GetInputState() Then DoEvents  '來代替DoEvents可使循環(huán)運(yùn)行更快

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

4.控時(shí)循環(huán)和變速齒輪
請(qǐng)看下面的代碼:
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
其中可用的代碼(除去加"'" 號(hào)的代碼)就是通常的控時(shí)循環(huán)代碼
運(yùn)行代碼并不會(huì)出現(xiàn)錯(cuò)誤,但在循環(huán)過程,請(qǐng)開啟變速齒輪看看
當(dāng)關(guān)閉齒輪時(shí),將發(fā)現(xiàn)text1.text停止了,別慌,等一段時(shí)間它又會(huì)繼續(xù)(這要看你設(shè)定的時(shí)間,這里是50毫秒,如果設(shè)定的太長(zhǎng)text1.text將半天都沒變化,這是怎么回事?
變速齒輪在啟動(dòng)時(shí)將hook.dll映射到你的程序地址運(yùn)行,更改了timegettime()函數(shù)獲取的時(shí)間
如果在句2和句3間插入debug.print timegettime,timegettime-kt 將發(fā)現(xiàn),在關(guān)閉齒輪的瞬間后者變成了負(fù)值,timegettime變小了,所以才造成需要等很久
如果是編寫游戲,而用戶開了齒輪,那可就慘了

解決方案:
a.用句3代替句2,這個(gè)方法最簡(jiǎn)便,雖然不符實(shí),但不會(huì)出問題,建議使用
b.不要句5,換用句6(這樣就能達(dá)到效果嗎?) 因?yàn)辇X輪還是從DoEvents語句運(yùn)行時(shí)才能插的進(jìn)來,所以只要kt=timegettime 和 timegettime之間沒有DoEvents就不會(huì)出錯(cuò)
ab.兩種方法都有些小問題,但無大礙DoEvents有興趣者請(qǐng)自己分析

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


該文章在 2014/3/13 8:11:55 編輯過
關(guān)鍵字查詢
相關(guān)文章
正在查詢...
點(diǎn)晴ERP是一款針對(duì)中小制造業(yè)的專業(yè)生產(chǎn)管理軟件系統(tǒng),系統(tǒng)成熟度和易用性得到了國內(nèi)大量中小企業(yè)的青睞。
點(diǎn)晴PMS碼頭管理系統(tǒng)主要針對(duì)港口碼頭集裝箱與散貨日常運(yùn)作、調(diào)度、堆場(chǎng)、車隊(duì)、財(cái)務(wù)費(fèi)用、相關(guān)報(bào)表等業(yè)務(wù)管理,結(jié)合碼頭的業(yè)務(wù)特點(diǎn),圍繞調(diào)度、堆場(chǎng)作業(yè)而開發(fā)的。集技術(shù)的先進(jìn)性、管理的有效性于一體,是物流碼頭及其他港口類企業(yè)的高效ERP管理信息系統(tǒng)。
點(diǎn)晴WMS倉儲(chǔ)管理系統(tǒng)提供了貨物產(chǎn)品管理,銷售管理,采購管理,倉儲(chǔ)管理,倉庫管理,保質(zhì)期管理,貨位管理,庫位管理,生產(chǎn)管理,WMS管理系統(tǒng),標(biāo)簽打印,條形碼,二維碼管理,批號(hào)管理軟件。
點(diǎn)晴免費(fèi)OA是一款軟件和通用服務(wù)都免費(fèi),不限功能、不限時(shí)間、不限用戶的免費(fèi)OA協(xié)同辦公管理系統(tǒng)。
Copyright 2010-2025 ClickSun All Rights Reserved