首先来看看如何创建线程:
Console.WriteLine(Process.GetCurrentProcess().Threads.Count); Thread t1 = new Thread(() => { Thread.Sleep(1000); Thread t = Thread.CurrentThread; Console.WriteLine("Name: " + t.Name); Console.WriteLine("ManagedThreadId: " + t.ManagedThreadId); Console.WriteLine("State: " + t.ThreadState); Console.WriteLine("Priority: " + t.Priority); Console.WriteLine("IsBackground: " + t.IsBackground); Console.WriteLine("IsThreadPoolThread: " + t.IsThreadPoolThread); }) { Name = "Thread1", Priority = ThreadPriority.Highest }; t1.Start(); Console.WriteLine(Process.GetCurrentProcess().Threads.Count);
我们在Thread的构造方法中传入一个Lambda表达式,对应ThreadStart委托(无参void返回值的方法)来构造一个线程任务。这段程序中有几个注意点:
1)从输出结果中可以看到,当前程序启动后就3三个线程,新开线程后显示为4个线程,在线程方法中休眠了一秒,防止主线程执行完次线程就过早结束了。UI框架示例分享
2)我们可以为线程设置一个名字,方便调试。我们也可以设置线程的优先级,这个在之后会有进一步介绍。
3)第7行,托管线程的唯一标识符,微软建议使用托管线程的Id而不是操作系统中线程的Id来跟踪线程。
4)第10行代码输出了当前线程不是后台线程,也就是是前台线程,这是默认值。进程会等待前台线程结束结束,而如果是后台线程的话,所有前台线程结束后后台线程自动终止。对于Windows GUI应用程序来说,使用后台线程很可能发生诡异现象,也就是在程序从任务管理器的应用程序一栏中消失后其进程还在,只能通过手动终止进程来释放内存。
5)第11行代码表明这个线程不是由线程池创建的,有关线程池见后文的介绍。
那么我们再来看看如何为线程传入参数,一种方式是使用匹配ParameterizedThreadStart委托(object参数void返回值)的方法:
new Thread((date) => Console.WriteLine(((DateTime)date).ToString())).Start(DateTime.Now);
由于参数是object类型的,我们在使用的时候不得不进行转换,而且还有一个问题就是不支持多个参数,如果要多个参数的话只能使用自定义的对象进行包装,我们还可以使用另外一种方法,那就是使用一个无参方法来包装线程方法主体:
new Thread(() => Add(1, 2)).Start();
static void Add(int i, int j) { Console.WriteLine(i + j); }
上述几行代码的运行结果如下:
再来看一下后台线程前台线程:
new Thread(() => Console.ReadLine()) { IsBackground = false }.Start();
这是默认情况,可以看到控制台一直在等待用户的输入,按回车后程序结束,如果把IsBackground属性设置为true的话,可以看到程序在运行后马上接结束了,并没有等待线程方法的结束。UI框架示例分享
之前说过线程的优先级属性,我们做一个实验:
bool b = true; new Thread(() => { while (b) { i++; } }) { Priority = ThreadPriority.Highest }.Start(); new Thread(() => { while (b) { j++; } }) { Priority = ThreadPriority.Lowest }.Start(); Thread.Sleep(1000); b = false; Console.WriteLine("i: {0}, j: {1}", i, j);
开启两个线程做的事情很简单,累加一个静态变量的值,一个优先级最高,一个优先级最低,然后让主线程等待1秒输出结果:
从结果中可以看到,优先级高的线程得到运行的次数比优先级低的线程多那么一点,但即使是最低优先级的线程都有很大的机会来执行。
现在再来看看线程的中断:
Thread t2 = new Thread(() => { try { while (true) { Console.WriteLine(Thread.CurrentThread.ThreadState); Thread.Sleep(1000); } } catch (ThreadAbortException abortException) { Console.WriteLine("catch"); Console.WriteLine(Thread.CurrentThread.ThreadState); Console.WriteLine((string)abortException.ExceptionState); } }); t2.Start(); Thread.Sleep(2000); t2.Abort("haha"); Thread.Sleep(100); Console.WriteLine(t2.ThreadState);
在线程方法中,我们1秒输出一次线程的状态,然后主线程休眠2秒后中断线程,略微等待一点时间,等线程中断结束后再获取一次线程的状态。可以看到:
每一秒出现一次Running,2秒后由于线程中断处罚ThreadAbortException进入catch块,此时线程的状态是AbortRequested,也能接受到我们中断线程时传入的状态信息,最后线程的状态为Stopped。UI框架示例分享
现在再来看看线程的Join,用于阻塞调用线程等Join的线程完成,或传入一个时间,阻塞一定的时间:
Thread t3 = new Thread(() => { for (int k = 0; k < 10; k++) { Thread.Sleep(100); Console.Write("X"); } Console.WriteLine(); }); Thread t4 = new Thread(() => { for (int k = 0; k < 10; k++) { Thread.Sleep(100); Console.Write("Y"); } Console.WriteLine(); }); t3.Start(); t3.Join(TimeSpan.FromMilliseconds(500)); t4.Start(); Console.WriteLine();
这里可以看到,启动t3之后,我们让主线程阻塞500毫秒,这样的话t3应该已经输出若干X了,然后我们启动t4,随后的500毫秒,t3和t4交替输出X和Y,最后500毫秒由于t3已经结束,所以只会输出Y:
最后,再来看一个有趣的问题:
我们设置一个静态字段:
static int threadstaticvalue;
然后创建两个线程来循环累加这个值:
new Thread(() => { for (int l = 0; l < 100000; l++) { threadstaticvalue++; } Console.WriteLine("from {0}: {1}", Thread.CurrentThread.Name, threadstaticvalue); }) { Name = "1" }.Start(); new Thread(() => { for (int m = 0; m < 200000; m++) { threadstaticvalue++; } Console.WriteLine("from {0}: {1}", Thread.CurrentThread.Name, threadstaticvalue); }) { Name = "2" }.Start();
运行几次输出结果如下:
虽然我们在代码中指定了两个线程分别累加值10万次和20万次,但是可以看到输出结果五花八门!这是因为两个线程都访问了共享的静态字段,可能错开访问可能正巧同步。其实,在静态字段上加上一个ThreadStatic特性就可以解决:资源分享
[ThreadStatic] static int threadstaticvalue;
线程同步这个话题很大,我们下次接着讨论。
相关推荐
### .NET下的多线程与并行计算:深入解析与应用 #### 一、引言 随着计算机硬件的发展,特别是多核处理器的普及,多线程和并行计算已成为现代软件开发不可或缺的一部分。本文旨在探讨.NET框架下多线程与并行计算的...
在.NET框架中,多线程和并行计算是提高应用程序性能和响应能力的关键技术。本文将深入探讨这两个概念,以及如何在.NET环境下有效地利用它们。 首先,多线程是指一个程序中存在多个执行流,每个执行流都有自己的独立...
在.NET框架中,多线程和并行计算是提高应用程序性能和响应能力的关键技术。本文将深入探讨这两个概念,以及如何在.NET环境下有效地利用它们。 首先,多线程是指一个程序中同时执行多个独立的线程,每个线程都有自己...
总之,这个.NET 2.0的多线程实例涵盖了多线程的创建与管理、线程同步、UI更新、文件系统操作、异步编程等多个核心知识点,对于理解和实践.NET环境下的并发编程具有很高的价值。通过学习和实践这些示例,开发者可以...
本示例主要介绍了如何使用`System.Threading.Thread`类来创建和管理多线程,以进行并行计算。 首先,我们需要引入必要的命名空间: ```csharp using System; using System.Collections.Generic; using System.Linq;...
在.NET框架中,C#语言提供了强大的多线程支持,使得开发者可以充分利用现代多核处理器的优势,实现并行处理和高效能编程。本资源包含六个C#.NET多线程的实例,涵盖了多线程的基本使用到更高级的概念,如线程互斥。...
通过理解和应用这些知识点,开发者能够熟练地在VB.NET中创建和管理多线程程序,优化应用程序的性能,解决并发问题,实现高效并行计算。在实际项目中,还需要根据具体需求和场景灵活运用多线程技术,确保程序的稳定性...
.NET Framework 4.0中引入的任务并行库(TPL)和并行LINQ(PLINQ)极大地简化了并行编程的过程,使得开发者可以更加专注于业务逻辑而不是底层的线程管理。随着多核处理器的普及以及对高性能计算需求的不断增长,并行...
在VB.NET编程中,多线程是一个至关重要的概念,它允许程序同时执行多个任务,提高应用程序的响应性和效率。在创建多线程应用时,尤其是对于UI(用户界面)密集型应用,可以避免因长时间计算导致的界面无响应,即防止...
ASP.NET多线程是.NET框架下进行并发编程的重要技术,尤其在开发高性能、高并发的Web应用程序时,熟练掌握多线程技术至关重要。本资源"ASP.NET多线程源码"提供了一组实践性的源代码,旨在帮助开发者深入理解并应用多...
.doc 格式 详细解析多线程技术 基础篇 • 怎样创建一个线程 • 受托管的线程与 Windows线程 • 前台线程与后台线程 • 名为BeginXXX和EndXXX的方法是做什么用的 • 异步和多线程有什么关联 WinForm多线程编程...
VB.net提供了多种机制来解决这些问题,包括`lock`语句、`Monitor`类、`Mutex`类等,它们可以帮助开发者确保代码在多线程环境下的正确性和稳定性。 #### 结论 多线程编程是提升软件性能和响应能力的关键技术,特别...
在编程领域,多线程是实现并发执行任务的关键技术,特别是在现代高性能计算中。VB.NET,作为Microsoft .NET框架的一部分,提供了丰富的支持来创建和管理多线程应用程序。本篇文章将深入探讨VB.NET多线程的基本概念、...
ASP.NET多线程技术是开发高效、响应迅速的Web应用程序的关键技术之一。它允许开发者在同一个应用程序中同时执行多个任务,提升系统性能并优化用户体验。本文将深入探讨ASP.NET多线程的基础概念、应用场景以及最佳...
异步编程可以避免线程切换的开销,但在某些场景下,多线程是提高并行性的有效手段。 ### WinForm多线程问题 在WinForm应用中,由于UI控件的访问必须在创建它们的线程(UI线程)中进行,因此会出现`...
在.NET框架中,C#语言提供了强大的多线程支持,使得开发者可以充分利用现代多核处理器的优势,实现并行处理和高效编程。以下是对标题和描述中提到的C#.NET多线程实例进行的详细解析: 1. **多线程基本使用**: 多...
C#面试题 包括 ADO.net 多线程等 C#面试题 包括 ADO.net 多线程等 C#面试题 包括 ADO.net 多线程等 C#面试题 包括 ADO.net 多线程等 C#面试题 包括 ADO.net 多线程等
.NET Framework 4 引入了一组强大的并行计算工具,称为`.NET Parallel`,它使得开发者可以轻松地利用多核处理器的优势,实现高效的多线程和高并发编程。这一技术的引入大大提升了应用程序的性能,特别是在处理大数据...
本实例集包含了六个关于C#.NET多线程的实战案例,涵盖了多线程的基本使用以及多线程间的互斥控制,旨在帮助开发者深入理解和熟练运用多线程技术。 一、多线程基本使用 多线程的基本使用包括创建线程、启动线程、...