vb.net线程详细讲解

VisualBasic作为易学易用的编程语言,为多少程序员所仰慕,VB历经多年的发展,从大家都熟知的VB6到如今的VB.NET,虽然这两个差别很大= =
本文主要讲述“线程”这么一个东西。

线程是程序中一个单一的顺序控制流程。在单个程序中同时运行多个线程完成不同的工作,称为多线程。


为什么使用线程?
 我们一般编写的程序代码总是从main函数(控制台),Sub New()(类的构造函数),Load(窗体加载)开始执行的,从上往下,执行每一个调用,有明确的先后顺序,一个sub或者function完成之后,才进行下一个sub或者function。的确,这样程序的逻辑性很强,每个过程排队,依次来。
 但是,这样必然会在一定程度上降低应用程序的运行效率,如果某个过程代码很长,所需要的时间长,那么程序在执行这个过程的时候会出现“假死”,停止响应用户操作,知道过程全部执行完毕。
 关于“假死”:我们编写的Windows应用程序有一个UI线程,用于接收和响应用户界面的操作。而我们编写的代码一般都是基于这个线程的,位于单一线程中的代码也是从上往下依次进行,所以当UI线程中某一过程花费的时间很长时,界面不再响应,因为它很忙,这时就出现了长时间的停顿,也就是“假死”,而用户会认为这很卡。
 因此,如果我们在UI线程的基础上另开一个线程,让代码分支执行的话,就不会卡了。但同时,我们又不得不面临这样一个问题:万一线程执行的过程,和UI执行的过程有冲突怎么办?当你在非UI线程中调用UI线程中的某一个控件,设置它的某某属性,这时你会收到这样一条错误:
 →线程间操作无效: 从不是创建控件“xxx”的线程访问它。
怎么办啊? 别急,后文会有解释。


废话不多说,读者跟我一起来。

首先,我们来体验一下使用线程带来的好处和问题。
1.打开VS(我的是2010版本)。
2.创建Windows窗体应用程序随便弄个名。
3.在窗体上放两个控件,Label1个Button1,如图,其他属性默认:

我们想实现这样一个功能:
在Label1上面动态显示数字,从0~9000,我们希望看到数字的变化。


创建如下代码:
Public Class Form1
 Private Sub Button1_Click(sender As System.Object, e As System.EventArgs) Handles Button1.Click
    test() 
 End Sub

 Private Sub test()
    For i = 0 To 9000
     Label1.Text = i.ToString
    Next 
 End Sub
End Class
意思就是点击按钮,进入test过程中,通过For循环,依次显示数字,真是这样吗?运行试试...
不出我意料的话,最后直接显示的是9000,中间还卡了一下。

后面的0都不在了。。
那么,我们要显示动态变化又怎么办呢?


《SD敢达大作战》正版卡牌巨作!  强力机师,超级敢达等你来战! 
怎么办呢?


把上面的代码修改一下,使用线程。
Imports System.Threading ’导入命名空间

Public Class Form1
Dim t As Thread ’定义全局线程变量

Private Sub Button1_Click(sender As System.Object, e As System.EventArgs) Handles Button1.Click
t = New Thread(AddressOf test) ’创建线程,使它指向test过程,注意该过程不能带有参数
t.Start() ’启动线程
End Sub

Private Sub test()
For i = 0 To 9000
Label1.Text = i.ToString
Next
t.Abort()
End Sub
End Class
再次运行,点击确定,出错啦?什么错?如图:

由于是从一个新的线程调用UI线程中窗体控件,所以这个做法很危险,你直接被拒绝了。
有一个解决办法,就是让编译器不进行跨线程检查。
在Load代码第一行加一句:
CheckForIllegalCrossThreadCalls = False

再次运行程序,就不会有错了,你还能看见动态变化,并且没有“假死”:

如上就是线程的好处。
但是:
CheckForIllegalCrossThreadCalls = False
一句跨线程调用Windows窗体控件就万能了吗?毕竟这种方式很不优秀。
不然,请看下文。


我们再创建一个窗体Form2,放一个Label0,如图:

我希望通过Form1的按钮,让Form2中的Label0显示0~9000.
代码如下:
Form1:

Imports System.Threading
Public Class Form1
  Dim t As Thread
  Private Sub Form1_Load(sender As Object, e As System.EventArgs) Handles Me.Load
   Form2.Show()'启动时显示Form2
End Sub

  Private Sub Button1_Click(sender As System.Object, e As System.EventArgs) Handles Button1.Click
   CheckForIllegalCrossThreadCalls = False
   t = New Thread(AddressOf test)
   t.Start()
  End Sub

  Private Sub test()
   For i = 0 To 9000
     Form2.Label0.Text = i.ToString '注意这里改了
   Next
   t.Abort()
  End Sub
End Class
//运行试试,咦?Form2里面怎么没变?
难道没有执行那句代码?
添加断点看看?

很明显执行了。但是就是没显示,程序不听话了?
CheckForIllegalCrossThreadCalls = False 
没辙了吧?
看后文。


跨两个UI调用,
CheckForIllegalCrossThreadCalls = False 
确实不太给力,那么如何是好?

这里,我们就要用到“委托”和invoke?什么东东啊? 往后看。。。
在Form1里面添加委托声明代码(带一个参数),和控件更新过程(带一个参数),
在test中使用Me.Invoke调用委托,执行UpdateUI,并向里面传一个参数i
稍微修改一下,其余代码不变:

下面运行试试?你看到了什么?是不是动态变化了哦?

删除这一句:CheckForIllegalCrossThreadCalls = False 也行。


收起回复
  • 10楼
  • 2012-01-25 14:11
    那么,像这样跨线程调用Windows窗体控件就实现了,并且这是被允许的安全方法。
    有了线程和委托的联合,我们就能创建更加人性化的程序了,快速而又安全。
    多线程的实现就是开很多线程罢了,记住最后一定要.Abort关闭线程,不然如果线程未结束,程序退出只是UI退出,线程还在呢....
    评论 2
    添加红包

    请填写红包祝福语或标题

    红包个数最小为10个

    红包金额最低5元

    当前余额3.43前往充值 >
    需支付:10.00
    成就一亿技术人!
    领取后你会自动成为博主和红包主的粉丝 规则
    hope_wisdom
    发出的红包
    实付
    使用余额支付
    点击重新获取
    扫码支付
    钱包余额 0

    抵扣说明:

    1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
    2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

    余额充值