我们都知道,在这个行业,追求的就是用最少的时间学最多的知识,这是我写这个系列最想达到的目标,在最快的时间内,帮助更多的人学习更多的线程知识。
前一篇,讲述了线程基础,给大家铺垫了一个基础,这一篇着重介绍线程的作用及其工作方式,顺便小试牛刀一把。
现在我想提出,最直接的问题是:
为什么要使用线程?
为什么要使用线程?答案只有三点(欢迎补充^_^):
- 使用线程可以将代码同其他代码隔离。这将提高应用程序的可靠性,这不仅仅是应用程序所需要的,更是Windows引入线程的真正原因。
- 使用线程可以简化编程。这个答案有利有弊,需要从两方面考虑:第一点:使用线程的同时也就意味着会付出一些资源作为代价,对于现在的计算机,付出资源是值得的,因为它的资源根本没有发掘出来。第二点:引用线程的时候是在需要相互协作的代码时,不能随便的引用线程,盲目的使用线程,只会增加代码复杂度。
- 可以用线程来程序的实现并发执行,双管齐下,效率,你懂的(∩_∩)。
说了这么些,总得试试手啊,使用线程?so easy:
1 class Program 2 { 3 static void Main(string[] args) 4 { 5 Console.WriteLine("主线程:启动专用线程..."); 6 Thread dedicatedThread = new Thread(StartCode); 7 8 dedicatedThread.Start(5); 9 10 11 Console.WriteLine("主线程:运行到此");12 Thread.Sleep(10000);//模拟主线程操作13 dedicatedThread.Join();14 Console.WriteLine("主线程:运行完毕");15 }16 17 private static void StartCode(Object obj)18 {19 Console.WriteLine("子线程:启动代码!{0}",obj);20 Thread.Sleep(1000);//模拟代码操作21 22 }23 }
下面是我两次运行的结果:
这是最基本的线程运用,好像不难哦?但是不提倡这么用。
中间有个Join方法,Join的作用是:造成调用线程阻塞当前执行的任何代码,直到dedicatedThread所代表的线程终止或销毁。
同一个程序出现两种输出结果,这是为什么?程序的每次输出不该是一样的吗?
两种输出的不一样是因为Windows对两个线程进行调度的方式不同,这无法控制。Windows抢占式多线程这一概念觉得了这因素。
这个知识点大家有必要了解,这个例子是个专用线程你可以这么用,但是建议应避免这么做,CLR的线程池可以更安全的完成这些事,如果你一定要创
建自己的线程,开始执行专用线程时,需考虑以下四点内容:
- 线程需要以非普通线程优先运行。线程池线程都是普通优先级运行,可以更改这个优先级,但不建议这么做。在不同的线程池操纵之间,优先级的更改是无法延续的(线程池这个概念下篇解析)。
- 需要线程表现为一个前台进程,防止应用程序在线程结束它的任务之前终止。(线程池的线程始终是后台线程,如果CLR要终止进程,它们就可能无法被迫完成任务)
- 一个计算限制的任务需要长时间的运行,就像例子中StartCode(),它执行的就是计算限制的任务。为长时间运行的任务创建一个专业线程,用于避免这个问题。
- 任务线程可能调用Abort()(属于Thread)来提前终止它。
线程可以分为前台线程和后台线程
CLR将每个线程要么视为前台,要么视为后台线程。当一个进程中的所有前台线程停止运行时(也就是我们按右上角的X,关闭程序),CLR将强制终止仍在运行的后台线程直接终止,不会异常。
基于这个原因,前台线程的使用应该用于执行确实想完成的任务,就用个我们正在用浏览器(下面称前台线程)做例子:
你正在浏览本篇内容,就意味着这个前台线程,正要完成你所需求的任务指令,解析HTML代码,便于你的阅读,这是首要任务。而加载书签,读取收藏网址的信息等,关键的后台功能,能在应用程序重启的时候继续执行,如果关闭前台线程,它们没必要保持活动的状态。
下面来看个前台线程和后台线程的程序实现:
1 class Program 2 { 3 static void Main(string[] args) 4 { 5 Thread t = new Thread(StartCode); 6 // t.IsBackground = true; 7 t.Start(); 8 9 Console.WriteLine("主线程运行完毕!");10 }11 12 private static void StartCode()13 {14 Console.WriteLine("开始执行子线程...");15 Thread.Sleep(10000);//模拟代码操作16 Console.WriteLine("子线程执行完毕!");17 18 }19 }
这段可运行代码就是默认模式,执行的前台代码。它的输出也是你能预测的:
在“开始执行子线程...”的时候,需要等待10秒。
去掉第6行的注释,再看看运行结果:
它不会等待,并看不到“子线程执行完毕”这句话。
前台的好处是,你可以保证你的后台线程能执行完毕,后台线程的好处是,你不用管它的执行。
在一个线程的生存期中,任何时候都可以进行前后台互换。
CLR要提供前台线程和后台线程的概念来更好的支持应用程序域(俗称AppDoMain),每个AppDoMain都可以运行一个单独的应用程序,每个应用程序都有它的前台线程,一个应用程序退出,前台线程终止,对应的后台线程也要终止,但CLR线程仍然需要运行,使其他应用程序继续运行,所有应用程序退出后,整个进程就可以销毁了。CLR算是线程运行的一个空间。
最后,说个大家十分熟悉的功能,用过Visual Studio 的开发人员,我保证你们都体验过这个功能。
智能提示都知道,这个是典型的线程运用,很快捷很舒心是不是?
当你写代码写到兴头时,发现编译器画出红线提示你,某处代码出错了。有没有发现这个?怎么实现的?
在你停止输入的时候,编译器就会开始编译代码,检测代码中的出错部分,很人性化,这大大提高了C#开发人员的工作效率,而在这,线程功不可没,线程的力量可见一斑。
线程基础还有一节重要的。下回讲~~~更新有点慢,是为了写出更好的博客。(^。^)y-~~