首页 > 代码库 > .Net 多线程编程由浅入深知识梳理
.Net 多线程编程由浅入深知识梳理
1.Using System.Threading;
多线程就是一个包工头,让手底下的十个工人去搬砖头的事儿。
------------不带任何传入参数的Thread
2.ThreadStart ts = new ThreadStart(某线程A的Func);
Thread th = new Thread(ts);
th.Start();
public void 某线程A的Func()
{
Console.WriteLine("线程A把事情做好了!");
Console.ReadKey();
}
原理解析
2.1 ThreadStart原型为一个委托类型:
public delegate void ThreadStart();
2.2 ThreadStart这个委托类型是没有任何传入参数的
如下可以让ThreadStart能够接收参数
public class MyThread
{
private double _width = 10.0;
private double _height = 9.0;
private double result = 0.0;
public MyThread(double width,double height)
{
this._width = width;
this._height = height;
}
public void MyThreadExc()
{
Console.WriteLine("MyThread Execute Start ...");
Thread.Sleep(2000);
result = _width * _height;
Console.WriteLine("MyThread Execute Done:The Result is {0}", result);
}
}
static void Main(string[] args)
{
MyThread th = new MyThread(5.0,3.0);
Thread t = new Thread(new ThreadStart(th.MyThreadExc));
t.Start();
Console.ReadKey();
}
------------带有参数的Thread
3.ParameterizedThreadStart pts = new ParameterizedThreadStart(某线程B的Func);
Thread th = new Thread(pts);
th.Start();
public void 某线程B的Func(object param)
{
Console.WriteLine("线程B接收了param并把事情做好了!");
}
原理解析
3.1 ParameterizedThreadStart原型为一个委托类型:
public delegate void ParameterizedThreadStart(object obj);
3.2 ParameterizedThreadStart这个委托类型是接收Object传参的
------------带有传入参数并且有返回参数
刚才说到主线程需要知道子线程什么时候执行完成,我们可以用Thread.ThreadState枚举来进行判断
当线程的ThreadState==Thread.Stop时,此时就可以判断该Thread已经结束,这时它的结果就可以使用了,如果不是这个状态,就继续执行别的工作,或等待再次尝试。倘若需要有多个子
线程返回,并且需要他们的返回内容再进行异步计算,这就叫做线程同步了,下面将介绍一种方法,能够自定义参数个数,并且返回数据,使用起来很方便
使用委托的异步调用方法和回调
1. 需要异步调用的Func定义为一个委托
2. BeginInvoke来异步调用 -- 第一个参数是传参,第二个参数是当线程执行完毕之后调用的方法 (穿插复习Invoke 与 BeginInvoke的区别)
delegate double MyJobDelegate(double d);
static MyJobDelegate myJob;
double jobResult=0.0;
static void Main(string[] args)
{
myJob = new MyJobDelegate(Job1);
myJob.BeginInvoke(5.09,new AsyncCallback(Job1Done),null)
}
static double Job1(double d)
{
return d*Math.PI;
}
static void Job1Done(IAsyncResult result)
{
Console.WriteLine(myJob.EndInvoke(result));
}
Asynchronous 异步
Synchronous 同步
[在C#中使用线程的方法很多,使用委托的BeginInvoke和EndInvoke方法就是其中之一。BeginInvoke方法可以使用线程异步地执行委托所指向的方法。然后通过EndInvoke方法获得方法的返回值
(EndInvoke方法的返回值就是被调用方法的返回值),或是确定方法已经被成功调用。]
-------------线程池
线程虽然是个好东西,但是也是一个资源消耗大户,我们需要用多线程,但是又不希望线程过多,线程池的机制就应运而生啦,.Net为我们提供了一个现成的ThreadPool,他的使用如下:
///
///首先下面这是线程池线程所要执行的回调方法
///
using System;
using System.Runtime.InteropServices;
namespace System.Threading
{
// 摘要:
// 表示线程池线程要执行的回调方法。
//
// 参数:
// state:
// 包含回调方法要使用的信息的对象。
[ComVisible(true)]
public delegate void WaitCallback(object state);
}
static WaitCallback wait;
static void Main(string[] args)
{
wait = new WaitCallback(myJob);
ThreadPool.QueueUserWorkItem(wait,1.0);
ThreadPool.QueueUserWorkItem(wait,2.0);
ThreadPool.QueueUserWorkItem(wait,3.0);
ThreadPool.QueueUserWorkItem(wait,4.0);
Console.ReadKey();
}
private static void myJob(object obj)
{
double d= (double)obj;
Console.WriteLine("Job Exec Result: {0}",d*Math.PI);
}
-------------控制权
场景:如果Winfrom程式主线程创建了一个控件TextBox1,随后某个子线程开始创建并且工作,产生一个返回值,如果此时将该返回值返回给TextBox1,程式就会报错:禁止跨线程访问控件;
那么此时我们就需要用到控件的一个方法:BeginInvoke或Invoke,这个方法可以把执行上下文切换回创建这个控件的线程:
delegate void ChangeTextBoxDelegate(string newtext);
private void Button1_Click(object sender,EventArgs e)
{
ParameterizedThreadStart pts = new ParameterizedThreadStart(myJob);
Thread th = new Thread(pts);
th.Start(1.09);
}
----------------方案1---------------
private void myJob(object obj)
{
this.BeginInvoke(new ChangeTextBoxDelegate(this.TextBox1.AppendText)),((double)obj * Math.PI).ToString());
}
或者 (推荐)
----------------方案2--------------
private void SetTextBox1Text(string text)
{
if(TextBox1.InvokeRequired)
{
ChangeTextBoxDelegate ct = SetTextBox1Text;
TextBox1.BeginInvoke(ct,text);
}
else
{
TextBox1.Text = text;
}
}
private void myJob(object obj)
{
SetTextBox1Text((double)obj * Math.PI).ToString());
}
参考:http://www.cnblogs.com/yizhu2000/archive/2007/10/12/922637.html
.Net 多线程编程由浅入深知识梳理