转:http://blog.csdn.net/zhoufoxcn/article/details/4402999

在实例化Thread的实例,需要提供一个委托,在实例化这个委托时所用到的参数是线程将来启动时要运行的方法。在.net中提供了两种启动线程的方式,一种是不带参数的启动方式,另一种是带参数的启动的方式。
 不带参数的启动方式
 如果启动参数时无需其它额外的信息,可以使用ThreadStart来实例化Thread,如下面的代码:

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using System.Threading;
  5. namespace StartThread
  6. {
  7. class Program
  8. {
  9. int interval = 200;
  10. static void Main(string[] args)
  11. {
  12. Program p = new Program();
  13. Thread nonParameterThread = new Thread(new ThreadStart(p.NonParameterRun));
  14. nonParameterThread.Start();
  15. }
  16. /// <summary>
  17. /// 不带参数的启动方法
  18. /// </summary>
  19. public void NonParameterRun()
  20. {
  21. for (int i = 0; i < 10; i++)
  22. {
  23. Console.WriteLine("系统当前时间毫秒值:"+DateTime.Now.Millisecond.ToString());
  24. Thread.Sleep(interval);//让线程暂停
  25. }
  26. }
  27. }

程序的运行效果我们不用运行也会知道,那就是在循环中将系统当前时间的毫秒部分输出出来,在每次输出之后会将当前线程暂停一下,直到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类型参数时,必须在启动方法中进行相应的类型转换。
 下面就是一个例子,在启动线程时指定了线程的暂停间隔,代码如下:

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using System.Threading;
  5. namespace StartThread
  6. {
  7. class Program
  8. {
  9. int interval = 200;
  10. static void Main(string[] args)
  11. {
  12. Program p = new Program();
  13. Thread parameterThread = new Thread(new ParameterizedThreadStart(p.ParameterRun));
  14. parameterThread.Name = "Thread A:";
  15. parameterThread.Start(30);
  16. }
  17. /// <summary>
  18. /// 带参数的启动方法
  19. /// </summary>
  20. /// <param name="ms">让线程在运行过程中的休眠间隔</param>
  21. public void ParameterRun(object ms)
  22. {
  23. int j = 10;
  24. int.TryParse(ms.ToString(), out j);//这里采用了TryParse方法,避免不能转换时出现异常
  25. for (int i = 0; i < 10; i++)
  26. {
  27. Console.WriteLine(Thread.CurrentThread.Name+"系统当前时间毫秒值:" + DateTime.Now.Millisecond.ToString());
  28. Thread.Sleep(j);//让线程暂停
  29. }
  30. }
  31. }
  32. }

在这个方法里,我们在启动线程时顺便指定了线程的暂停间隔,也就是这句:
 parameterThread.Start(30);
 线程启动时运行的方法是public void ParameterRun(object ms),这个值为30的int类型变量被装箱成object,所以在方法中还需要将它转换成int类型,这个可以通过拆箱或者其它办法解决。
 假如我们要启动两个线程,每个线程的暂停间隔不一样,启动代码如下:

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using System.Threading;
  5. namespace StartThread
  6. {
  7. class Program
  8. {
  9. int interval = 200;
  10. static void Main(string[] args)
  11. {
  12. Program p = new Program();
  13. Thread parameterThread = new Thread(new ParameterizedThreadStart(p.ParameterRun));
  14. parameterThread.Name = "Thread A:";
  15. parameterThread.Start(30);
  16. //启动第二个线程
  17. parameterThread = new Thread(new ParameterizedThreadStart(p.ParameterRun));
  18. parameterThread.Name = "Thread B:";
  19. parameterThread.Start(60);
  20. }
  21. /// <summary>
  22. /// 带参数的启动方法
  23. /// </summary>
  24. /// <param name="ms">让线程在运行过程中的休眠间隔</param>
  25. public void ParameterRun(object ms)
  26. {
  27. int j = 10;
  28. int.TryParse(ms.ToString(), out j);//这里采用了TryParse方法,避免不能转换时出现异常
  29. for (int i = 0; i < 10; i++)
  30. {
  31. Console.WriteLine(Thread.CurrentThread.Name+"系统当前时间毫秒值:" + DateTime.Now.Millisecond.ToString());
  32. Thread.Sleep(j);//让线程暂停
  33. }
  34. }
  35. }
  36. }

对上面的代码做一点说明,就是线程启动之后,线程的实例不必再存在,例如在上面的代码中我用的是同一个实例实例化了两个线程,并且这两个线程运行很正常。
 
 继续探索
 上面解决了一个问题,如果在启动线程时需要参数如何解决,如果针对上面的问题继续发掘,比如:在启动线程时不但要指定线程的暂停间隔,还需要指定循环次数(在上面的所有例子中都是执行10次的),这个问题该如何解决呢?
 有两种办法可以解决:
 首先可以继续在ParameterizedThreadStart这里做文章,因为这里可以使用一个Object类型的参数,那么可以通过数组或者一个类来解决(因为它们都是Object的子类)。我在做某个系统时确实采用数组处理过这种情况,这样就要求在线程启动方法中必须清楚知道数组中每个参数的用途,不是太方便。
 这里说说重新定义一个实体类来解决的方法,代码如下。

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using System.Threading;
  5. namespace StartThread
  6. {
  7. class MyThreadParameter
  8. {
  9. private int interval;
  10. private int loopCount;
  11. /// <summary>
  12. /// 循环次数
  13. /// </summary>
  14. public int LoopCount
  15. {
  16. get { return loopCount; }
  17. }
  18. /// <summary>
  19. /// 线程的暂停间隔
  20. /// </summary>
  21. public int Interval
  22. {
  23. get { return interval; }
  24. }
  25. /// <summary>
  26. /// 构造函数
  27. /// </summary>
  28. /// <param name="interval">线程的暂停间隔</param>
  29. /// <param name="loopCount">循环次数</param>
  30. public MyThreadParameter(int interval,int loopCount)
  31. {
  32. this.interval = interval;
  33. this.loopCount = loopCount;
  34. }
  35. }
  36. class Program
  37. {
  38. int interval = 200;
  39. static void Main(string[] args)
  40. {
  41. Program p = new Program();
  42. Thread parameterThread = new Thread(new ParameterizedThreadStart(p.MyParameterRun));
  43. parameterThread.Name = "Thread A:";
  44. MyThreadParameter paramter = new MyThreadParameter(50, 20);
  45. parameterThread.Start(paramter);
  46. }
  47. /// <summary>
  48. /// 带多个参数的启动方法
  49. /// </summary>
  50. /// <param name="ms">方法参数</param>
  51. public void MyParameterRun(object ms)
  52. {
  53. MyThreadParameter parameter = ms as MyThreadParameter;//类型转换
  54. if (parameter != null)
  55. {
  56. for (int i = 0; i < parameter.LoopCount; i++)
  57. {
  58. Console.WriteLine(Thread.CurrentThread.Name + "系统当前时间毫秒值:" + DateTime.Now.Millisecond.ToString());
  59. Thread.Sleep(parameter.Interval);//让线程暂停
  60. }
  61. }
  62. }
  63. }
  64. }

第二种方法和上面方法有些相似,也是需要引入外部类,并且将Thread实例放在引入的类中,这种情况适合于在线程中处理的业务逻辑比较复杂的情况。在前不久处理的一个项目中我用过这种情况,它是用来实现双向数据传输的。
 如果实现上面的效果,代码如下:

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Text;
  4. using System.Threading;
  5. namespace StartThread
  6. {
  7. class MyThreadParameter
  8. {
  9. private int interval;
  10. private int loopCount;
  11. private Thread thread;
  12. /// <summary>
  13. /// 构造函数
  14. /// </summary>
  15. /// <param name="interval">线程的暂停间隔</param>
  16. /// <param name="loopCount">循环次数</param>
  17. public MyThreadParameter(int interval,int loopCount)
  18. {
  19. this.interval = interval;
  20. this.loopCount = loopCount;
  21. thread = new Thread(new ThreadStart(Run));
  22. }
  23. public void Start()
  24. {
  25. if (thread != null)
  26. {
  27. thread.Start();
  28. }
  29. }
  30. private void Run()
  31. {
  32. for (int i = 0; i < loopCount; i++)
  33. {
  34. Console.WriteLine("系统当前时间毫秒值:" + DateTime.Now.Millisecond.ToString());
  35. Thread.Sleep(interval);//让线程暂停
  36. }
  37. }
  38. }
  39. class Program
  40. {
  41. static void Main(string[] args)
  42. {
  43. MyThreadParameter parameterThread = new MyThreadParameter(30, 50);
  44. parameterThread.Start();
  45. }
  46. }
  47. }

上面的代码的运行效果和前面的代码运行效果类似,只不过是将业务处理代码放在一个单独的类MyThreadParameter中,使得MyThreadParameter看起来也像一个Thread,实际上维护的还是其内部的Thread,在一些大型系统中这样做的好处是便于维护。

C#多线程编程(1):线程的启动的更多相关文章

  1. .NET面试题解析(07)-多线程编程与线程同步

      系列文章目录地址: .NET面试题解析(00)-开篇来谈谈面试 & 系列文章索引 关于线程的知识点其实是很多的,比如多线程编程.线程上下文.异步编程.线程同步构造.GUI的跨线程访问等等, ...

  2. .NET面试题解析(07)-多线程编程与线程同步 (转)

    http://www.cnblogs.com/anding/p/5301754.html 系列文章目录地址: .NET面试题解析(00)-开篇来谈谈面试 & 系列文章索引 关于线程的知识点其实 ...

  3. Python中的多线程编程,线程安全与锁(二)

    在我的上篇博文Python中的多线程编程,线程安全与锁(一)中,我们熟悉了多线程编程与线程安全相关重要概念, Threading.Lock实现互斥锁的简单示例,两种死锁(迭代死锁和互相等待死锁)情况及 ...

  4. C#多线程编程实例 线程与窗体交互

    C#多线程编程实例 线程与窗体交互 代码: public partial class Form1 : Form { //声明线程数组 Thread[] workThreads = ]; public ...

  5. vc 基于对话框多线程编程实例——线程之间的通信

     vc基于对话框多线程编程实例——线程之间的通信 实例:

  6. Python中的多线程编程,线程安全与锁(一)

    1. 多线程编程与线程安全相关重要概念 在我的上篇博文 聊聊Python中的GIL 中,我们熟悉了几个特别重要的概念:GIL,线程,进程, 线程安全,原子操作. 以下是简单回顾,详细介绍请直接看聊聊P ...

  7. Win32多线程编程(2) — 线程控制

    Win32线程控制只有是围绕线程这一内核对象的创建.挂起.恢复.终结以及通信等操作,这些操作都依赖于Win32操作系统提供的一组API和具体编译器的C运行时库函数.本篇围绕这些操作接口介绍在Windo ...

  8. Win32多线程编程(3) — 线程同步与通信

      一.线程间数据通信 系统从进程的地址空间中分配内存给线程栈使用.新线程与创建它的线程在相同的进程上下文中运行.因此,新线程可以访问进程内核对象的所有句柄.进程中的所有内存以及同一个进程中其他所有线 ...

  9. Delphi 实现多线程编程的线程类 TThread

    http://blog.csdn.net/henreash/article/details/3183119 Delphi中有一个线程类TThread是用来实现多线程编程的,这个绝大多数Delphi书藉 ...

  10. Java多线程编程(3)--线程安全性

    一.线程安全性   一般而言,如果一个类在单线程环境下能够运作正常,并且在多线程环境下,在其使用方不必为其做任何改变的情况下也能运作正常,那么我们就称其是线程安全的.反之,如果一个类在单线程环境下运作 ...

随机推荐

  1. asp.net web 开发登录相关操作的控件LoginName、LoginStatus和LoginView控件使用详解

    http://book.51cto.com/art/200909/154039.htm http://book.51cto.com/art/200909/154041.htm ASP.NET提供了一套 ...

  2. 【DataStructure In Python】Python模拟二叉树

    使用Python模拟二叉树的基本操作,感觉写起来很别扭.最近做编译的优化,觉得拓扑排序这种东西比较强多.近期刷ACM,发现STL不会用实在太伤了.决定花点儿时间学习一下STL.Boost其实也很强大. ...

  3. DataProvider 传递参数

    package roger.testng; import org.testng.annotations.DataProvider; import org.testng.annotations.Test ...

  4. CentOS6.4下Git服务器Gitosis安装配置

    1.安装GIt: #yum install git 2.增加一个git用户 #useradd git #passwd git 3.创建git仓库存储目录,设置权限 #mkdir /home/git/r ...

  5. C#学习:集合、迭代、泛型(1)

    一.System.Collections名称空间下几个接口表征着集合的功能: 1.IEnumerable:表征着迭代功能 public interface IEnumerable { IEnumera ...

  6. 2D游戏编程7—星空案例

    // INCLUDES /////////////////////////////////////////////// #define WIN32_LEAN_AND_MEAN // just say ...

  7. leetcode之Palindrome Partitioning

    方法一:DFS递归,判断每一个是否为回文数 1,首先要有一个判断字符串是否是回文的函数.容易实现,字符串从两边同时往中间走,看字符是否相同; 2,深度优先搜索思想对字符串进行遍历.得到结果.例如,s ...

  8. NOIP2015 提高组(senior) 解题报告

    过了这么久才来发解题报告,蒟蒻实在惭愧 /w\ Day1 T1 [思路] 模拟 [代码] #include<iostream> #include<cstring> #inclu ...

  9. 算法导论学习-heapsort

    heap的定义:如果数组a[1,....n]满足:a[i]>a[2*i] && a[i]>a[2*i+1],1<=i<=n/2,那么就是一个heap,而且是ma ...

  10. POSIX-Centos查看rpm包安装位置及相关信息

    rpm -qifl `which svnserve`; http://www.ctohome.com/FuWuQi/56/122.html [国外服务器及ip租用价格] Centos6 kvm网桥配置 ...