2)我们可以用更好的方式使用.NET的委托和线程池
A)委托
WaitCallback 委托,表示线程池线程要执行的回调方法。
WaitCallback 表示要在 ThreadPool 线程上执行的回调方法。 创建委托,方法是将回调方法传递给 WaitCallback 构造函数。 您的方法必须具有此处所显示的签名。
通过将 WaitCallback 委托传递给 ThreadPool.QueueUserWorkItem 来将任务排入队列以便执行。 您的回调方法将在某个线程池线程可用时执行。
如果要将信息传递给回调方法,请创建包含所需信息的对象,并在将任务排入队列以便执行时将它传递给 QueueUserWorkItem。 每次执行您的回调方法时,state 参数都包含此对象。
B)线程池
ThreadPool 类提供一个线程池,该线程池可用于执行任务、发送工作项、处理异步 I/O、代表其他线程等待以及处理计时器。
许多应用程序创建的线程都要在休眠状态中消耗大量时间,以等待事件发生。 其他线程可能进入休眠状态,只被定期唤醒以轮询更改或更新状态信息。 线程池通过为应用程序提供一个由系统管理的辅助线程池,使您可以更为有效地使用线程。
可以将与等待操作不相关的工作项排列到线程池。 若要请求由线程池中的一个线程来处理工作项,请调用 QueueUserWorkItem 方法。 此方法将对将被从线程池中选定的线程调用的方法或委托的引用用作参数。 一个工作项排入队列后就无法再取消它。
计时器队列中的计时器以及已注册的等待操作也使用线程池。 它们的回调函数也会排列到线程池。
每个进程都有一个线程池。 从 .NET Framework 4 版开始,进程的线程池的默认大小由虚拟地址空间的大小等多个因素决定。 进程可以调用 GetMaxThreads 方法以确定线程的数量。 使用 SetMaxThreads 方法可以更改线程池中的线程数。 每个线程使用默认的堆栈大小并按照默认的优先级运行。
c)代码如下:
Imports System Imports System.Threading Imports System.Diagnostics Imports System.Diagnostics.ThreadState Module Module1 <MTAThread()> _ Sub Main() '完成任务的线程号 Dim finishedid As Integer '定义WaitCallback 委托 Dim mywaitcallback(4) As WaitCallback '定义线程传参的对象数组 Dim calculateifno(4) As CalculateIfno '分别给每个线程创建AutoResetEvent对象 Dim threadevent(4) As AutoResetEvent Console.WriteLine(Now.ToLongTimeString & "线程对象创建完毕,开始执行线程") '设置线程执行前相关参数,并执行线程,将参数对象传入线程 For i = 0 To threadevent.GetUpperBound(0) '设置参数对象 threadevent(i) = New AutoResetEvent(False) calculateifno(i) = New CalculateIfno calculateifno(i).threadevent = threadevent(i) calculateifno(i).result = 0 calculateifno(i).threadname = i & "号线程" '在线程池中加入线程,将参数对象传入线程执行 mywaitcallback(i) = New WaitCallback(AddressOf mythreadrun) ThreadPool.QueueUserWorkItem(mywaitcallback(i), calculateifno(i)) Next '等待其中一个线程完成累加,然后将其它未完成任务的线程终止 finishedid = WaitHandle.WaitAny(threadevent) '线程执行完毕 Console.WriteLine(Now.ToLongTimeString & " " & finishedid & "号线程完成任务,计算完毕!" & Environment.NewLine & "计算结果为:" & calculateifno(finishedid).result) End Sub Public Class CalculateIfno Private _threadevent As AutoResetEvent Private _result As Long Private _threadname As String Public Property threadevent As AutoResetEvent Get Return _threadevent End Get Set(ByVal value As AutoResetEvent) _threadevent = value End Set End Property Public Property result As Long Get Return _result End Get Set(ByVal value As Long) _result = value End Set End Property Public Property threadname As String Get Return _threadname End Get Set(ByVal value As String) _threadname = value End Set End Property End Class Public Sub mythreadrun(ByVal calculateifno As Object) Dim mynum As Integer Dim jg As Long = 0 Try For mynum = 1 To 1000 jg += mynum Thread.Sleep(5) Next Console.WriteLine(CType(calculateifno, CalculateIfno).threadname & " " & Now.ToLongTimeString & "线程运行完毕!") Catch Console.WriteLine(CType(calculateifno, CalculateIfno).threadname & " " & Now.ToLongTimeString & "线程异常终止!") '终止线程 Thread.CurrentThread.Abort() Finally CType(calculateifno, CalculateIfno).threadevent.Set() CType(calculateifno, CalculateIfno).result = jg End Try End Sub End Module
D)分析
1.参数类中增加了threadname ,存放线程名称,因为
调用 QueueUserWorkItem 方法。 此方法将对将被从线程池中选定的线程调用的方法或委托的引用用作参数。 一个工作项排入队列后就无法再取消它。并且没有提供直接指定线程池被调用的线程的名字的办法
Public Class CalculateIfno
Private _threadevent As AutoResetEvent
Private _result As Long
Private _threadname As String
Public Property threadevent As AutoResetEvent
Get
Return _threadevent
End Get
Set(ByVal value As AutoResetEvent)
_threadevent = value
End Set
End Property
Public Property result As Long
Get
Return _result
End Get
Set(ByVal value As Long)
_result = value
End Set
End Property
Public Property threadname As String
Get
Return _threadname
End Get
Set(ByVal value As String)
_threadname = value
End Set
End Property
End Class
2.QueueUserWorkItem方法