找回密码
 立即注册

QQ登录

只需一步,快速开始

扫一扫,访问微社区

查看: 381|回复: 0

[VBA程序]:如何通过按一个命令按钮事件来退出do循环?

[复制链接]
发表于 2004-7-16 10:44:17 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能,让你轻松玩转社区。

您需要 登录 才可以下载或查看,没有账号?立即注册

×
请教一个问题

一个do...loop循环中,可在内部设置exit do语句退出循环,如果要通过按一个命令按钮事件来退出do循环,应该怎么做?

发现一篇文章,用的是doevents语句



  1. VB6.0
  2.     Windows98
  3.     我要实现一个类似Windows复制(移动)文件时的提示窗体,耗时很长,且要求中断后能继续未完成的操作,不知使用DoEvents从长循环中跳出后,程序将从何处开始继续执行,是否是从DoEvents所在的Sub开始?(倔小孩)  
  4.    
  5.      事实上仅使用DoEvents,并不意味着从长循环中跳出。DoEvents只是允许用户选择其他按钮而已,不中断循环,不管用户如何操作,都继续执行DoEvents后面的语句,即使用户按下了中断按钮,你的循环仍然在继续,甚至你关闭了窗体,程序仍然继续在后台运行。正确的中断处理是这样的:
  6.     1、建立一个全局或窗体变量bRun。
  7.     2、在启动循环前设置bRun为True。
  8.     bRun = True
  9.     While bRun And (....)
  10.      ....
  11.      DoEvents
  12.     Wend
  13.     ...
  14.     3、在中断按钮Click事件中加入代码:
  15.     bRun = False
  16.     4、在Form_Unload事件中加入代码:
  17.     bRun = False
  18.     这样一旦用户按下了中断按钮,bRun = False,循环的条件就不满足了,所以退出循环,执行后续语句。你也可以采用下面的方式:
  19.     bRun = True
  20.     While ....
  21.      If Not bRun Then
  22.      Exit Sub
  23.      End If
  24.      ....
  25.      DoEvents
  26.     Wend
  27.     ...
  28.      
  29.     郭勇的意见:
  30.     《解析 事件,Doevents,闲置循环和控时循环中的难点问题》
  31.     Visualbasic6.0 代码
  32.     copyright guoyong in cqums(2004-2-26)
  33.     关键问题归于doevents 函数
  34.     DoEvents函数的功能是:转让控制权,以便让操作系统处理其它的事件。
  35.     问:为什么要用doevents?
  36.      A.在需要用某一循环处理相当耗时或者很快速的代码时,就需要用到它,以便用户能在起处理过程中能做其他事情,即程序能被控制,而不是无响应状态
  37.      B.vb6.0中多线程vb代码极度不稳定,而且无法调试,所以vb中的多线程用的很少(注:是指vb的代码在多线程中运行时不稳定)
  38.      C.timer控件可以起到后台运行作用,但其是通过事件控制,一是不稳定,二是速度太慢,如果想用其处理高速又耗系统的代码更本不能达到预期的效果
  39.      
  40.     下面将其某些用法和难点简介如下:
  41.      
  42.     (注: '** 后面的代码表示如果在该处用了这个语句
  43.     以下代码中用到了一些api函数,请用vb附带的api浏览器查阅)
  44.      
  45.     一. 基本用法:
  46.     1.窗体启动时如果要处理的事务太多或者用sleep函数暂停,造成其很久都不能出现时怎么办?
  47.     例如代码:
  48.     Private Sub Form_Load()
  49.     Show
  50.     '**DoEvents
  51.     Sleep 5000
  52.     End Sub
  53.     通常容易想到在sleep前加个show,但还是不能达到预想的效果,窗体虽然出来了,但好象只达到了一半,如果加上第3句,将看到效果大不相同
  54.      
  55.     2.如果有个很耗时的循环导致程序不响应,怎么办?
  56.     例如:
  57.     Dim L As Long
  58.     For L = 1 To 1000000
  59.     '** DoEvents
  60.     Next L
  61.     如果无'**,在循环过程中程序无法处理事件,对于用户来说是不响应,无法控制的
  62.     3.想在循环中看到处理过程?
  63.     同样:
  64.     Dim L As Long
  65.     For L = 1 To 10000
  66.     '** DoEvents
  67.     Text1.Text=Cstr(l)
  68.     Next L
  69.     无'** 时将无法看到text1中的变化,而只在循环结束时看到最后结果
  70.     4.怎样中止循环?
  71.     如果有:
  72.     Private Sub Command3_Click()
  73.     Dim L As Long
  74.     Do
  75.      L = L + 1
  76.      Debug.Print L
  77.      DoEvents
  78.     Loop
  79.     End Sub
  80.     会发现当关闭窗口后,debug中的数据仍然在变化,说明并没结束
  81.     需要如下:
  82.     Dim IsExit As Boolean
  83.     Private Sub Command1_Click()
  84.     Dim L As Long
  85.      IsExit = False
  86.      Do While DoEvents
  87.      If IsExit = True Then Exit Do
  88.      L = L + 1
  89.      Loop
  90.      
  91.     End Sub
  92.      
  93.     Private Sub Command2_Click()''或者在form_unload模块中等等
  94.      IsExit = True
  95.     End Sub
  96.     其中 isexit是全局变量
  97.     <>有些人喜欢用end语句来结束程序,小程序固然可以,但当太大,或者调用了某些特殊的api函数后可能导致预想不到的错误,如果装载了许多东西在程序结束时不处理将卸载很慢,而且这种做法也极不符合正规软件的要求...总之end语句毛病很多,此不详谈,建议少使用甚至不使用
  98.      
  99.     二. 其基本用法大概就这些,现在解析其中的一些[难点]
  100.      
  101.     1.为什么还是不能结束?
  102.     代码如下:
  103.     Dim IsExit As Boolean
  104.     Private Sub Command1_Click()
  105.     Dim L As Long
  106.      IsExit = False
  107.      Do
  108.      If IsExit = True Then Exit Do '句0
  109.      DoEvents '** 句1
  110.      Text1.Text = CStr(L) '** 句2
  111.      L = L + 1
  112.      Loop
  113.     End Sub
  114.      
  115.     Private Sub Form_Load()
  116.     Static N As Long
  117.     N = N + 1
  118.     MsgBox N
  119.     End Sub
  120.      
  121.     Private Sub Form_Unload(Cancel As Integer)
  122.      IsExit = True
  123.     End Sub
  124.      
  125.     运行结果:启动时msg显示1,点击command1,text1在变化
  126.     此时再点form右上角的小差(关闭窗体),发现vb运行控制上的按扭并没变化,说明程序还在运行.如果编译成程序后运行,按下ctrl+del+alt也可发现它还没结束.
  127.     通过读代码,并没发现错误,怎么回事?
  128.      
  129.     关键在于 句2 访问了控件的属性
  130.     代码运行路径:当在doevents 时,程序释放控制权,可以接收事件消息,form-unload事件只能从此处产生,假设此时关闭form ,unload事件发生,即doevents后就运行unload代码,得到isexit=t,并且form卸载,代码返回到doevents 之后,运行 句2.注意现在form 已经卸载了,text1从哪里来呢?
  131.     于是form重新装载,代码跳到form_load模块运行,所以在关闭窗体后可以看到msg 显示2,此模块运行完后再继续句2后面的代码,当下次循环遇到 句0时退出循环
  132.     另:既然退出了循环,怎么还不能结束?
  133.      vb程序规定(其实其他的windows语言一样):窗体卸载时并不是立即卸载其模块代码,而只先卸载窗体中的控件和一些属性值,程序中最后一个窗体卸载时才完全卸载.
  134.     在这个单窗体程序中,form卸载时因为循环的控制无法卸载代码,失去了卸载代码的机会,导致再也不能卸载(因为没卸载代码,所以运行的 句2 是并不会出错)
  135.     另:既然再次运行了form_load代码,怎么看不见窗体?
  136.      因为程序启动时窗体的到显示的消息,而只运行此模块并没有(如果在msgbox n语句前加上show,就可以看到它了)
  137.     如何解决?
  138.     通过以上分析,应该很简单,把句1 和句2调换一下就可以了,关键:
  139.     <仔细分析代码是如何运行的,避免在form已经卸载了情况下访问控件>
  140.      
  141.     2.用了doevents速度太慢了怎么办?
  142.      doevents的代价是速度变慢,但要程序响应又不得不用
  143.     其实doevents语句允许任何应用程序执行相关事件,而不仅仅是你自己的程序,所以变得很慢.
  144.     可以让它响应本程序事件动作,需要用到api函数GetInputState
  145.     例如用: If GetInputState() Then DoEvents '来代替doevents可使循环运行更快
  146.      
  147.     3.既要同时响应事件又要控件不变化,怎么办?
  148.     例如在一个长的循环中向listview控件中添加记录,无doevents时程序无响应,但有它时控件又闪的厉害
  149.     解决办法:a.不一定每次循环都doevents,可以在适当时间时才用,至少没那么闪
  150.     b.应用api函数 ValidateRect 功能是使指定的矩型区域生效,通知Windows不对指定的区域进行重画 另:InvalidateRect 功能相反,同时需要用到函数 GetClientRect 取得指定对象的矩形区域 应用*rect函数指定listview的矩形区不重画,即可避免闪烁(但还是要注意恢复重画,否则看不见了真实效果)
  151.      
  152.     4.控时循环和变速齿轮
  153.     请看下面的代码:
  154.     Option Explicit
  155.     Private Declare Function timeGetTime Lib "winmm.dll" () As Long
  156.     Dim IsExit As Boolean
  157.     Private Sub Command1_Click()
  158.     Dim L As Long
  159.     Dim Kt As Long
  160.     IsExit = False
  161.     Do
  162.      Kt = timeGetTime()
  163.      'do something
  164.      L = L + 1
  165.      Text1.Text = L
  166.      'DoEvents '句 1
  167.      While timeGetTime - Kt < 50 '句 2
  168.      'While Abs(timeGetTime - Kt) < 50 '句 3
  169.      'While Abs(timeGetTime - Kt) And (Not IsExit) < 50 '句 4
  170.      
  171.      DoEvents '句 5
  172.      Wend
  173.      'DoEvents '句 6
  174.      If IsExit Then Exit Do
  175.     Loop
  176.     End Sub
  177.      
  178.     Private Sub Form_Unload(Cancel As Integer)
  179.     IsExit = True
  180.     End Sub
  181.     其中可用的代码(除去加"'" 号的代码)就是通常的控时循环代码
  182.     运行代码并不会出现错误,但在循环过程,请开启变速齿轮看看
  183.     当关闭齿轮时,将发现text1.text停止了,别慌,等一段时间它又会继续(这要看你设定的时间,这里是50毫秒,如果设定的太长text1.text将半天都没变化,这是怎么回事?
  184.      变速齿轮在启动时将hook.dll映射到你的程序地址运行,更改了timegettime()函数获取的时间
  185.     如果在句2和句3间插入debug.print timegettime,timegettime-kt 将发现,在关闭齿轮的瞬间后者变成了负值,timegettime变小了,所以才造成需要等很久
  186.     如果是编写游戏,而用户开了齿轮,那可就惨了
  187.      
  188.     解决方案:
  189.     a.用句3代替句2,这个方法最简便,虽然不符实,但不会出问题,建议使用
  190.     b.不要句5,换用句6(这样就能达到效果吗?) 因为齿轮还是从doevents语句运行时才能插的进来,所以只要kt=timegettime 和 timegettime之间没有doevents就不会出错
  191.     ab.两种方法都有些小问题,但无大碍,有兴趣者请自己分析
  192.      
  193.     5.程序怎么"死了"?
  194.     这只是一些人编写时没注意到的小问题,提醒一下:
  195.     同样用上面的代码,如果设定的时间太短,以至在代码运行到句2时已经超时了,句5将不能运行了,当然程序就死了哦,以防万一,加上句1,所以此时也只能用a方案来解决齿轮的问题了
  196.     有必要用句4代替句3 吗? 除非你设定的时间太长,人家想关闭你的程序要等上好半天
  197.       


论坛插件加载方法
发帖求助前要善用【论坛搜索】功能,那里可能会有你要找的答案;
如果你在论坛求助问题,并且已经从坛友或者管理的回复中解决了问题,请把帖子标题加上【已解决】;
如何回报帮助你解决问题的坛友,一个好办法就是给对方加【D豆】,加分不会扣除自己的积分,做一个热心并受欢迎的人!
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|申请友链|Archiver|手机版|小黑屋|辽公网安备|晓东CAD家园 ( 辽ICP备15016793号 )

GMT+8, 2024-11-23 08:52 , Processed in 0.266147 second(s), 32 queries , Gzip On.

Powered by Discuz! X3.5

© 2001-2024 Discuz! Team.

快速回复 返回顶部 返回列表