使用 Visual Basic .NET 进行多线程编程(转自MSDN)续
发表于:2007-06-30来源:作者:点击数:
标签:
同步线程 同步在多线程编程的非结构化性质与同步处理的结构化次序之间提供了一个折衷的办法。 使用同步技术,可以完成以下操作: 在必须以特定顺序执行任务时,显式控制代码运行的次序。 - 或者 - 当两个线程同时共享相同的资源时,避免可能出现的问题。 例
同步线程
同步在多线程编程的非结构化性质与同步处理的结构化次序之间提供了一个折衷的办法。
使用同步技术,可以完成以下操作:
- 在必须以特定顺序执行任务时,显式控制代码运行的次序。
- 或者 - - 当两个线程同时共享相同的资源时,避免可能出现的问题。
例如,可以使用同步使显示过程处于等待状态,直至在另一线程中运行的数据检索过程结束。
同步的方法有两种:轮询和使用同步对象。轮询反复从循环中检查异步调用的状态。使用轮询管理线程的效率最低,因为反复检查各种线程属性的状态会浪费大量资源。
例如,如果轮询要查看线程是否已结束,可以使用
IsAlive 属性。使用此属性时要很小心,因为活动的线程不一定正在运行。可以使用线程的
ThreadState 属性来获得有关线程状态的详细信息。由于在任意给定时间,线程都可能处于多种状态,因此
ThreadState 中存储的值可以是
System.Threading.Threadstate 枚举中的值的组合。因此,在轮询时应当仔细检查所有相关的线程状态。例如,如果线程的状态表明它没有运行,则该线程可能已经完成。另一方面,它也可能被挂起或处于休眠状态。
可以想象,轮询为控制运行线程的次序,牺牲了多线程的部分优点。为此,可以使用效率较高的
Join 方法来控制线程。
Join 使调用过程处于等待状态,直至线程完成或调用超时(如果指定了超时)。“Join”这个名称来自这一想法,即创建的新线程是执行路径的一个分支。使用
Join 可以再次将单独的执行路径合并成一个线程。
图 1:线程
有一点需要清楚:
Join 是同步调用或阻塞调用。调用 Join 或等待句柄的等待方法后,调用过程将停止并等待线程发出信号通知它已经完成。Sub JoinThreads() Dim Thread1 As New System.Threading.Thread(AddressOf SomeTask) Thread1.Start() Thread1.Join() @# 等待线程运行结束。 MsgBox("线程运行结束")End Sub
这些控制线程的简单方法在管理少量线程时非常有用,但不适合大型项目。下一节将讨论可用于同步线程的一些高级技术。
高级同步技术
多线程应用程序通常使用等待句柄和监视器对象来同步多个线程。下表介绍了可用于同步线程的部分 .NET 框架类。
类用途
AutoResetEvent | 等待句柄,用于通知一个或多个等待线程发生了一个事件。AutoResetEvent 在等待线程被释放后自动将状态更改为已发出信号。 |
Interlocked | 为多个线程共享的变量提供原子操作。 |
ManualResetEvent | 等待句柄,用于通知一个或多个等待线程发生了一个事件。手动重置事件的状态将保持为已发出信号,直至 Reset 方法将其设置为未发出信号状态。同样,该状态将保持为未发出信号,直至 Set 方法将其设置为已发出信号状态。当对象的状态为已发出信号时,任意数量的等待线程(即通过调用一个等待函数开始对指定事件对象执行等待操作的线程)都可以被释放。 |
Monitor | 提供同步访问对象的机制。Visual Basic .NET 应用程序调用 SyncLock 以使用监视器对象。 |
Mutex | 等待句柄,可用于进程间同步。 |
ReaderWriterLock | 定义用于实现单个写入者和多个读取者的锁定。 |
Timer | 提供按指定间隔运行任务的机制。 |
WaitHandle | 封装操作系统特有的、等待对共享资源进行独占访问的对象。 |
等待句柄
等待句柄是将一个线程的状态通知另一个线程的对象。线程可以使用等待句柄,通知其他线程它们需要对资源进行独占访问。然后,其他线程必须等到没有线程在使用等待句柄时才能使用此资源。等待句柄有两种状态:已发出信号和未发出信号。不属于任何线程的等待句柄处于已发出信号状态。属于某线程的等待句柄处于未发出信号状态。
线程通过调用一种等待方法(例如
WaitOne、
WaitAny 或
WaitAll)来请求等待句柄的所有权。等待方法是与单独线程的
Join 方法相类似的阻塞调用。
- 如果没有其他线程拥有该等待句柄,则调用将立即返回 True,等待句柄的状态将更改为未发出信号,而拥有等待句柄的线程将继续运行。
- 如果线程调用了等待句柄的一种等待方法,但该等待句柄归另一线程所有,则调用线程将等待指定的时间(如果指定了超时),或者无限期地等待(未指定超时),直至其他线程释放等待句柄。如果指定了超时,并且在超时到期前释放等待句柄,则调用返回 True。否则,调用返回 False,并且进行调用的线程将继续运行。
拥有等待句柄的线程在运行结束后,或不再需要等待句柄时将调用
Set 方法。其他线程通过调用
Reset 方法,或者调用
WaitOne、
WaitAll 或
WaitAny 以及成功地等待某一线程调用
Set 方法之后,可以将等待句柄的状态重置为未发出信号。在单个等待线程被释放后,系统将
AutoResetEvent 句柄自动重置为未发出信号。如果没有线程处于等待状态,则事件对象的状态将保持为已发出信号。
方法用途
WaitOne | 接受一个等待句柄作为参数,并使调用线程处于等待状态,直至另一个进程调用 Set 将当前的等待句柄设置为已发出信号。 |
WaitAny | 接受一个等待句柄数组作为参数,并使调用线程处于等待状态,直至任一指定的等待句柄已通过调用 Set 设置为已发出信号。 |
WaitAll | 接受一个等待句柄数组作为参数,并使调用线程处于等待状态,直至所有指定的等待句柄已通过调用 Set 设置为已发出信号。 |
Set | 将指定的等待句柄的状态设置为已发出信号,并使任何等待线程继续运行。 |
Reset | 将指定事件的状态设置为未发出信号。 |
Visual Basic .NET 常用的等待句柄有三种:互斥对象、
ManualResetEvent 和
AutoResetEvent。后两种通常称为同步事件。
互斥对象
互斥对象是一次只能由一个线程拥有的同步对象。实际上,“互斥”这个名称来自互斥对象的所有权相互排斥这一事实。如果线程要对资源进行独占访问,则需要请求互斥对象的所有权。由于在任何时刻,只能有一个线程拥有互斥对象,因此其他线程必须等待,直至获得互斥对象的所有权后才能使用资源。
WaitOne 方法使调用线程等待获得互斥对象的所有权。如果拥有互斥对象的线程正常终止,则互斥对象的状态将设置为已发出信号,下一个等待线程将获得所有权。
同步事件
同步事件用于通知其他线程某件事情已发生或某个资源已可用。不要被这些使用“事件”一词的项误导。同步事件与其他 Visual Basic 事件不同,它们实际上是等待句柄。与其他等待句柄类似,同步事件也有两种状态:已发出信号和未发出信号。调用同步事件的一种等待方法的线程必须等待,直至另一个线程通过调用
Set 方法向事件发出通知。有两种同步事件类。线程使用
Set 方法将
ManualResetEvent 实例的状态设置为已发出信号。线程使用
Reset 方法,或者在控制返回到一个等待
WaitOne 的调用时,将
ManualResetEvent 实例的状态设置为未发出信号。还可以使用
Set 将
AutoResetEvent 类的实例设置为已发出信号,但是只要等待线程被通知事件已发出信号,这些实例就自动返回到未发出信号状态。
以下示例使用
AutoResetEvent 类来同步线程池任务。Sub StartTest() Dim AT As New AsyncTest() AT.StartTask()End SubClass AsyncTest Private Shared AsyncOpDone As New _ System.Threading.AutoResetEvent(False) Sub StartTask() Dim Tpool As System.Threading.ThreadPool Dim arg As String = "SomeArg" Tpool.QueueUserWorkItem(New System.Threading.WaitCallback( _ AddressOf Task), arg) @# 将一个任务排队。 AsyncOpDone.WaitOne() @# 等待线程调用 Set。 MsgBox("线程运行结束。") End Sub Sub Task(ByVal Arg As Object) MsgBox("线程正在启动。") System.Threading.Thread.Sleep(4000) @# 等待 4 秒钟。 MsgBox("状态对象包含字符串 " & CStr(Arg)) AsyncOpDone.Set() @# 通知线程运行结束。 End SubEnd Class
监视器对象和 SyncLock
监视器对象用于确保代码块在运行时不会被其他线程运行的代码中断。换句话说,直到同步代码块中的代码运行结束后,其他线程中的代码才能运行。在 Visual Basic .NET 中,
SyncLock 关键字用于简化对监视器对象的访问。在 Visual C#® .NET 中则使用
Lock 关键字。
例如,假设有一个反复异步读取数据并显示结果的程序。如果操作系统使用抢占式多任务处理技术,则可以中断正在运行的线程而将时间用于运行其他某个线程。如果不进行同步,则如果在显示数据时,代表数据的对象被其他线程修改,则可能会看到被部分更新的数据。
SyncLock 语句可以保证代码段在运行时不会被中断。以下示例说明了如何使用
SyncLock 为显示过程提供数据对象的独占访问权限。Class DataObject Public ObjText As String Public ObjTimeStamp As DateEnd ClassSub RunTasks() Dim MyDataObject As New DataObject() ReadDataAsync(MyDataObject) SyncLock MyDataObject DisplayResults(MyDataObject) End SyncLockEnd SubSub ReadDataAsync(ByRef MyDataObject As DataObject) @# 添加代码以异步读取和处理数据。End SubSub DisplayResults(ByVal MyDataObject As DataObject) @# 添加代码以显示结果。End Sub
如果需要确保代码段不会被在其它线程中运行的代码中断,请使用
SyncLock。
Interlocked 类
为避免在多个线程尝试同时更新或比较相同的值时可能出现的问题,可以使用
Interlocked 类的方法。此类的方法使您能够
安全地递增、递减、交换和比较任何线程中的值。以下示例说明了如何使用
Increment 方法来递增由在其它线程中运行的过程所共享的变量。Sub ThreadA(ByRef IntA As Integer) System.Threading.Interlocked.Increment(IntA)End SubSub ThreadB(ByRef IntA As Integer) System.Threading.Interlocked.Increment(IntA)End Sub
ReaderWriter 锁定
在某些情况下,可能希望只在写入数据时锁定资源,而在不更新数据时则允许多个客户端同时读取数据。
ReaderWriterLock 类在线程修改资源时强制独占访问资源,但在读取资源时允许进行非独占访问。ReaderWriter 锁定是独占锁定的一个很有用的替代选择,因为独占锁定使其他线程一直处于等待状态,即使那些线程并不需要更新数据。以下示例说明了如何使用 ReaderWriter 来协调多个线程的读写操作。Class ReadWrite@# 可以从多个线程中安全地调用@# ReadData 和 WriteData 方法。 Public ReadWriteLock As New System.Threading.ReaderWriterLock() Sub ReadData() @# 此过程从某个来源读取信息。 @# 读取锁定禁止在线程完成读取之前写入数据, @# 同时允许其他线程调用 ReadData。 ReadWriteLock.A
cquireReaderLock(System.Threading.Timeout.Infinite) Try @# 此处执行读取操作。 Finally ReadWriteLock.ReleaseReaderLock() @# 释放读取锁定。 End Try End Sub Sub WriteData() @# 此过程将信息写入某个来源。 @# 写入锁定禁止在线程完成写入操作前 @# 读取或写入数据。 ReadWriteLock.AcquireWriterLock(System.Threading.Timeout.Infinite) Try @# 此处执行写入操作。 Finally ReadWriteLock.ReleaseWriterLock() @# 释放写入锁定。 End Try End SubEnd Class
死锁
线程同步在多线程应用程序中十分重要,但在多个线程相互等待时总是存在死锁的危险。就象四个方向上都停有汽车的情况,每个人都在等待另一个人走,死锁使一切操作终止。显然,避免死锁非常重要。有许多情况会导致死锁,同样,避免死锁的方法也很多。虽然本文没有足够篇幅来讨论与死锁相关的所有问题,但有一点很重要,即认真规划是避免死锁的关键。在开始编码之前,通过图解多线程应用程序,通常可以预测死锁。
线程计时器
Threading.Timer 类对在单独线程中定期运行任务十分有用。例如,可以使用线程计时器检查
数据库的状态和完整性,或者备份重要文件。以下示例每两秒钟启动一个任务,并使用标志来启动使计时器停止的
Dispose 方法。本例将状态发送到输出窗口,因此在
测试代码之前,应按 CONTROL+ALT+O 键以使此窗口可见。Class StateObjClass@# 用于保留调用 TimerTask 所需的参数 Public SomeValue As Integer Public TimerReference As System.Threading.Timer Public TimerCanceled As BooleanEnd ClassSub RunTimer() Dim StateObj As New StateObjClass() StateObj.TimerCanceled = False StateObj.SomeValue = 1 Dim TimerDelegate As New Threading.TimerCallback(AddressOf TimerTask) @# 创建每隔 2 秒钟调用过程的计时器。 @# 注意:这里没有 Start 方法;创建实例之后, @# 计时器就开始运行。 Dim TimerItem As New System.Threading.Timer(TimerDelegate, StateObj, _ 2000, 2000) StateObj.TimerReference = TimerItem @# 为 Dispose 保存一个引用。 While StateObj.SomeValue < 10 @# 运行 10 个循环。 System.Threading.Thread.Sleep(1000) @# 等待 1 秒钟。 End While StateObj.TimerCanceled = True @# 请求计时器对象的 Dispose。End SubSub TimerTask(ByVal StateObj As Object) Dim State As StateObjClass = CType(StateObj, StateObjClass) Dim x As Integer @# 使用 Interlocked 类递增计数器变量。 System.Threading.Interlocked.Increment(State.SomeValue) De
bug.WriteLine("已启动了新线程 " & Now) If State.TimerCanceled Then @# 已请求 Dispose。 State.TimerReference.Dispose() Debug.WriteLine("完成时间 " & Now) End IfEnd Sub
当
System.Windows.Forms.Timer 类不可用时(例如在
开发控制台应用程序时),线程计时器特别有用。
取消任务
多线程的一个优点是,应用程序的用户界面部分始终可以作出响应,即使其他线程正在执行任务。同步事件和作为标志的字段通常用于通知其他线程停止。以下示例使用同步事件来取消任务。要使用本示例,请在项目中添加以下模块。要启动线程,请调用 StartCancel.StartTask() 方法。要取消一个或多个正在运行的线程,请调用 StartCancel.CancelTask() 方法。Module StartCancel Public CancelThread As New System.Threading.ManualResetEvent(False) Public ThreadisCanceled As New System.Threading.ManualResetEvent(False) Private Sub SomeLongTask() Dim LoopCount As Integer Dim Loops As Integer = 10 @# 在 While 循环中运行 10 秒钟代码,或者 @# 直至设置了 CancelThread。 While Not CancelThread.WaitOne(0, False) And LoopCount < Loops @# 此处执行某种类型的任务。 System.Threading.Thread.Sleep(1000) @# 休眠 1 秒钟。 LoopCount += 1 End While If CancelThread.WaitOne(0, False) Then @# 确认设置了 ManualResetEvent CancelThread。 ThreadisCanceled.Set() MsgBox("取消线程") Else MsgBox("线程运行结束") End If End Sub Public Sub StartTask() @# 启动新线程。 Dim th As New System.Threading.Thread(AddressOf SomeLongTask) CancelThread.Reset() ThreadisCanceled.Reset() th.Start() MsgBox("线程已启动") End Sub Public Sub CancelTask() @# 停止任何由 StartTask 过程启动的线程。 @# 注意,此线程同时接收和发送 @# 同步事件以协调线程操作。 CancelThread.Set() @# 设置 CancelThread 以通知线程停止。 If ThreadisCanceled.WaitOne(4000, False) Then @# 最多等待 4 秒钟,以便线程 @# 确认它已经停止。 MsgBox("线程已停止。") Else MsgBox("线程无法停止。") End If End SubEnd Module
总结
多线程处理是开发具有快速响应且可扩展的应用程序的关键。Visual Basic .NET 支持强大的多线程开发模型,使用此模型,开发人员可以快速利用多线程应用程序的强大功能。
- Visual Basic .NET 使用新的 .NET 框架类,可以方便地创建多线程应用程序。
- 请记住,虽然多线程可以提高性能,但每个线程都需要额外的内存来创建线程,还需要处理器时间来运行线程。
- 线程的属性和方法控制线程之间的交互,并确定正在运行的线程何时可以使用资源。
- 尽管多线程可能会造成混乱,但可以使用同步技术来控制运行的线程。
- 多线程通过高效地分配可用资源,提高了应用程序的可扩展性,但也增加了应用程序的复杂性。
使用本文介绍的技术可以开发出非常专业的应用程序,从而处理即使是最耗用处理器的任务。
原文转自:http://www.ltesting.net