C#多线程编程(1):线程的启动
转:http://blog.csdn.net/zhoufoxcn/article/details/4402999
在实例化Thread的实例,需要提供一个委托,在实例化这个委托时所用到的参数是线程将来启动时要运行的方法。在.net中提供了两种启动线程的方式,一种是不带参数的启动方式,另一种是带参数的启动的方式。
不带参数的启动方式
如果启动参数时无需其它额外的信息,可以使用ThreadStart来实例化Thread,如下面的代码:
- using System;
- using System.Collections.Generic;
- using System.Text;
- using System.Threading;
- namespace StartThread
- {
- class Program
- {
- int interval = 200;
- static void Main(string[] args)
- {
- Program p = new Program();
- Thread nonParameterThread = new Thread(new ThreadStart(p.NonParameterRun));
- nonParameterThread.Start();
- }
- /// <summary>
- /// 不带参数的启动方法
- /// </summary>
- public void NonParameterRun()
- {
- for (int i = 0; i < 10; i++)
- {
- Console.WriteLine("系统当前时间毫秒值:"+DateTime.Now.Millisecond.ToString());
- Thread.Sleep(interval);//让线程暂停
- }
- }
- }
程序的运行效果我们不用运行也会知道,那就是在循环中将系统当前时间的毫秒部分输出出来,在每次输出之后会将当前线程暂停一下,直到10次之后运行完毕,终止线程的执行。
在上面的代码中我们是通过定义全局变量的方法来指定线程暂停间隔,按照这种方法,假如要运行10个线程,每个线程的暂停间隔不一样的话,就需要定义10个全局变量,虽然最终不影响系统的运行效果,但是总觉得不是太爽。
有没有比较简单一点的办法呢?有!那就是使用带参数的启动方法。
带参数的启动方法
如果要在实例化线程时要带一些参数,就不能用ThreadStart委托作为构造函数的参数来实例化Thread了,而要ParameterizedThreadStart委托,和ThreadStart一样的是它也是线程启动时要执行的方法,和ThreadStart不同的是,它在实例化时可以用一个带有一个Object参数的方法作为构造函数的参数,而实例化ThreadStart时所用到的方法是没有参数的。
为什么是Object这样的参数呢?很简单,因为在.net中Object是所有类型的基类,用它可以表示Array(数组)、Interface(接口)、ValueType(值类型,如bool,byte,char,short,int,float,long,double等)、class(类)等.net中的类型。当然,这也意味着如果你要启动一个线程,给它传递一个int类型参数时,必须在启动方法中进行相应的类型转换。
下面就是一个例子,在启动线程时指定了线程的暂停间隔,代码如下:
- using System;
- using System.Collections.Generic;
- using System.Text;
- using System.Threading;
- namespace StartThread
- {
- class Program
- {
- int interval = 200;
- static void Main(string[] args)
- {
- Program p = new Program();
- Thread parameterThread = new Thread(new ParameterizedThreadStart(p.ParameterRun));
- parameterThread.Name = "Thread A:";
- parameterThread.Start(30);
- }
- /// <summary>
- /// 带参数的启动方法
- /// </summary>
- /// <param name="ms">让线程在运行过程中的休眠间隔</param>
- public void ParameterRun(object ms)
- {
- int j = 10;
- int.TryParse(ms.ToString(), out j);//这里采用了TryParse方法,避免不能转换时出现异常
- for (int i = 0; i < 10; i++)
- {
- Console.WriteLine(Thread.CurrentThread.Name+"系统当前时间毫秒值:" + DateTime.Now.Millisecond.ToString());
- Thread.Sleep(j);//让线程暂停
- }
- }
- }
- }
在这个方法里,我们在启动线程时顺便指定了线程的暂停间隔,也就是这句:
parameterThread.Start(30);
线程启动时运行的方法是public void ParameterRun(object ms),这个值为30的int类型变量被装箱成object,所以在方法中还需要将它转换成int类型,这个可以通过拆箱或者其它办法解决。
假如我们要启动两个线程,每个线程的暂停间隔不一样,启动代码如下:
- using System;
- using System.Collections.Generic;
- using System.Text;
- using System.Threading;
- namespace StartThread
- {
- class Program
- {
- int interval = 200;
- static void Main(string[] args)
- {
- Program p = new Program();
- Thread parameterThread = new Thread(new ParameterizedThreadStart(p.ParameterRun));
- parameterThread.Name = "Thread A:";
- parameterThread.Start(30);
- //启动第二个线程
- parameterThread = new Thread(new ParameterizedThreadStart(p.ParameterRun));
- parameterThread.Name = "Thread B:";
- parameterThread.Start(60);
- }
- /// <summary>
- /// 带参数的启动方法
- /// </summary>
- /// <param name="ms">让线程在运行过程中的休眠间隔</param>
- public void ParameterRun(object ms)
- {
- int j = 10;
- int.TryParse(ms.ToString(), out j);//这里采用了TryParse方法,避免不能转换时出现异常
- for (int i = 0; i < 10; i++)
- {
- Console.WriteLine(Thread.CurrentThread.Name+"系统当前时间毫秒值:" + DateTime.Now.Millisecond.ToString());
- Thread.Sleep(j);//让线程暂停
- }
- }
- }
- }
对上面的代码做一点说明,就是线程启动之后,线程的实例不必再存在,例如在上面的代码中我用的是同一个实例实例化了两个线程,并且这两个线程运行很正常。
继续探索
上面解决了一个问题,如果在启动线程时需要参数如何解决,如果针对上面的问题继续发掘,比如:在启动线程时不但要指定线程的暂停间隔,还需要指定循环次数(在上面的所有例子中都是执行10次的),这个问题该如何解决呢?
有两种办法可以解决:
首先可以继续在ParameterizedThreadStart这里做文章,因为这里可以使用一个Object类型的参数,那么可以通过数组或者一个类来解决(因为它们都是Object的子类)。我在做某个系统时确实采用数组处理过这种情况,这样就要求在线程启动方法中必须清楚知道数组中每个参数的用途,不是太方便。
这里说说重新定义一个实体类来解决的方法,代码如下。
- using System;
- using System.Collections.Generic;
- using System.Text;
- using System.Threading;
- namespace StartThread
- {
- class MyThreadParameter
- {
- private int interval;
- private int loopCount;
- /// <summary>
- /// 循环次数
- /// </summary>
- public int LoopCount
- {
- get { return loopCount; }
- }
- /// <summary>
- /// 线程的暂停间隔
- /// </summary>
- public int Interval
- {
- get { return interval; }
- }
- /// <summary>
- /// 构造函数
- /// </summary>
- /// <param name="interval">线程的暂停间隔</param>
- /// <param name="loopCount">循环次数</param>
- public MyThreadParameter(int interval,int loopCount)
- {
- this.interval = interval;
- this.loopCount = loopCount;
- }
- }
- class Program
- {
- int interval = 200;
- static void Main(string[] args)
- {
- Program p = new Program();
- Thread parameterThread = new Thread(new ParameterizedThreadStart(p.MyParameterRun));
- parameterThread.Name = "Thread A:";
- MyThreadParameter paramter = new MyThreadParameter(50, 20);
- parameterThread.Start(paramter);
- }
- /// <summary>
- /// 带多个参数的启动方法
- /// </summary>
- /// <param name="ms">方法参数</param>
- public void MyParameterRun(object ms)
- {
- MyThreadParameter parameter = ms as MyThreadParameter;//类型转换
- if (parameter != null)
- {
- for (int i = 0; i < parameter.LoopCount; i++)
- {
- Console.WriteLine(Thread.CurrentThread.Name + "系统当前时间毫秒值:" + DateTime.Now.Millisecond.ToString());
- Thread.Sleep(parameter.Interval);//让线程暂停
- }
- }
- }
- }
- }
第二种方法和上面方法有些相似,也是需要引入外部类,并且将Thread实例放在引入的类中,这种情况适合于在线程中处理的业务逻辑比较复杂的情况。在前不久处理的一个项目中我用过这种情况,它是用来实现双向数据传输的。
如果实现上面的效果,代码如下:
- using System;
- using System.Collections.Generic;
- using System.Text;
- using System.Threading;
- namespace StartThread
- {
- class MyThreadParameter
- {
- private int interval;
- private int loopCount;
- private Thread thread;
- /// <summary>
- /// 构造函数
- /// </summary>
- /// <param name="interval">线程的暂停间隔</param>
- /// <param name="loopCount">循环次数</param>
- public MyThreadParameter(int interval,int loopCount)
- {
- this.interval = interval;
- this.loopCount = loopCount;
- thread = new Thread(new ThreadStart(Run));
- }
- public void Start()
- {
- if (thread != null)
- {
- thread.Start();
- }
- }
- private void Run()
- {
- for (int i = 0; i < loopCount; i++)
- {
- Console.WriteLine("系统当前时间毫秒值:" + DateTime.Now.Millisecond.ToString());
- Thread.Sleep(interval);//让线程暂停
- }
- }
- }
- class Program
- {
- static void Main(string[] args)
- {
- MyThreadParameter parameterThread = new MyThreadParameter(30, 50);
- parameterThread.Start();
- }
- }
- }
上面的代码的运行效果和前面的代码运行效果类似,只不过是将业务处理代码放在一个单独的类MyThreadParameter中,使得MyThreadParameter看起来也像一个Thread,实际上维护的还是其内部的Thread,在一些大型系统中这样做的好处是便于维护。
C#多线程编程(1):线程的启动的更多相关文章
- .NET面试题解析(07)-多线程编程与线程同步
系列文章目录地址: .NET面试题解析(00)-开篇来谈谈面试 & 系列文章索引 关于线程的知识点其实是很多的,比如多线程编程.线程上下文.异步编程.线程同步构造.GUI的跨线程访问等等, ...
- .NET面试题解析(07)-多线程编程与线程同步 (转)
http://www.cnblogs.com/anding/p/5301754.html 系列文章目录地址: .NET面试题解析(00)-开篇来谈谈面试 & 系列文章索引 关于线程的知识点其实 ...
- Python中的多线程编程,线程安全与锁(二)
在我的上篇博文Python中的多线程编程,线程安全与锁(一)中,我们熟悉了多线程编程与线程安全相关重要概念, Threading.Lock实现互斥锁的简单示例,两种死锁(迭代死锁和互相等待死锁)情况及 ...
- C#多线程编程实例 线程与窗体交互
C#多线程编程实例 线程与窗体交互 代码: public partial class Form1 : Form { //声明线程数组 Thread[] workThreads = ]; public ...
- vc 基于对话框多线程编程实例——线程之间的通信
vc基于对话框多线程编程实例——线程之间的通信 实例:
- Python中的多线程编程,线程安全与锁(一)
1. 多线程编程与线程安全相关重要概念 在我的上篇博文 聊聊Python中的GIL 中,我们熟悉了几个特别重要的概念:GIL,线程,进程, 线程安全,原子操作. 以下是简单回顾,详细介绍请直接看聊聊P ...
- Win32多线程编程(2) — 线程控制
Win32线程控制只有是围绕线程这一内核对象的创建.挂起.恢复.终结以及通信等操作,这些操作都依赖于Win32操作系统提供的一组API和具体编译器的C运行时库函数.本篇围绕这些操作接口介绍在Windo ...
- Win32多线程编程(3) — 线程同步与通信
一.线程间数据通信 系统从进程的地址空间中分配内存给线程栈使用.新线程与创建它的线程在相同的进程上下文中运行.因此,新线程可以访问进程内核对象的所有句柄.进程中的所有内存以及同一个进程中其他所有线 ...
- Delphi 实现多线程编程的线程类 TThread
http://blog.csdn.net/henreash/article/details/3183119 Delphi中有一个线程类TThread是用来实现多线程编程的,这个绝大多数Delphi书藉 ...
- Java多线程编程(3)--线程安全性
一.线程安全性 一般而言,如果一个类在单线程环境下能够运作正常,并且在多线程环境下,在其使用方不必为其做任何改变的情况下也能运作正常,那么我们就称其是线程安全的.反之,如果一个类在单线程环境下运作 ...
随机推荐
- MIME(Multipurpose Internet Mail Extensions)多用途互联网邮件扩展类
MIME(Multipurpose Internet Mail Extensions)多用途互联网邮件扩展类型.是设定某种扩展名的文件用一种应用程序来打开的方式类型,当该扩展名文件被访问的时候,浏览器 ...
- 抽离CodeIgniter的数据库访问类 可以独立使用
好吧,因为组织需要,最近又开始转战php了,业务逻辑都还好说,主要是老大要求在数据访问层上加上登录态验证.其实这种要求也是合理的,互联网服务要求上层保护下层,但下层不能完全相信上层.但是问题也就来了, ...
- Bootstrap Paginator分页插件的使用
今天,我为大家带来的一款做得非常优秀的分页插件BootStrap Paginator,他是一款js插件,由于本人也是才刚刚搞出来的,所以暂时对它也不是特别了解,只能大楖告诉大家怎么使用.我这里使用的是 ...
- Android自定义组合控件
今天和大家分享下组合控件的使用.很多时候android自定义控件并不能满足需求,如何做呢?很多方法,可以自己绘制一个,可以通过继承基础控件来重写某些环节,当然也可以将控件组合成一个新控件,这也是最方便 ...
- __cdecl、__stdcall、__fastcall、thiscall 进栈、出栈区别
https://en.wikipedia.org/wiki/X86_calling_conventions https://msdn.microsoft.com/en-us/library/984x0 ...
- android手机屏幕分辨率 及 sp dip(dp) px 区别 及高中低分辨率时处理
分辨率,是指单位长度内包含的像素点的数量,它的单位通常为像素/英寸(ppi).以分辨率为1024×768的屏幕来说,即每一条水平线上包含有1024个像素点,共有768条线,即扫描列数为1024列,行数 ...
- 使用CPU探查器优化XAML程序
如果您正在开发一个使用 XAML (是否是 c + +. C# 或 VB) 的 Windows 商店应用程序,还有一个好的机会来提高应用程序的性能.为了帮助完成这一点,我们所有在售的能够应用开发 Wi ...
- HDU-3622 Bomb Game 2sat
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3622 题意:一个平面上有很多的炸弹,每个炸弹的爆炸范围是一样的,求最大的爆炸范围使得炸弹之间不相互影响 ...
- Java 线程池架构原理和源码解析(ThreadPoolExecutor)
在前面介绍JUC的文章中,提到了关于线程池Execotors的创建介绍,在文章:<java之JUC系列-外部Tools>中第一部分有详细的说明,请参阅: 文章中其实说明了外部的使用方式,但 ...
- jenkins持续集成配置备忘
配过好几次,但是老忘记.记录备忘. 安装下列插件: ssh和git插件用来下载源码 publish over ssh 插件用来发布代码到测试环境. email插件用来更强大的email配置. file ...