首页 > 技术文章 > ASP.NET本质论阅读----线程与异步

ruanyifeng 2015-09-22 16:23 原文

线程

概要:操作系统通过线程对程序的执行进行管理

线程的数据结构:1.线程的核心对象(包含线程的当前寄存器状态),调用线程时-寄存器状态被加载到CPU中。

        2.线程环境块TEB:一块用户模式下的内存

        3.用户模式的堆栈:用户程序的局部变量和参数的存储

        4.内核模式的堆栈:用于访问操作系统时的堆栈

线程分类:前台线程与后台线程。pS:所有前台线程终止,程序终止。

线程执行任务分类:工作者线程和I/O线程

         工作者线程:计算密集的任务,CPU与线程资源充分利用

         I/O线程:CPU可以不必参与处理过程

线程池

线程的创建与启动

  由于线程用来表示代码的执行,所以需要通过委托进行,常见的3种线程委托

  1.ThreadStart委托

  public  delegate void ThreadStart();

  2.ParameterizedThreadStart委托

  public delegate void ParameterizedThreadStart(Object obj)

  3.TimerCallback委托

  Timer类实现

  public Timer(TimerCallback    //时间到时执行的委托

        ,State        //传递的参数

        ,DueTime      //创建到第一次执行的时差

        ,Period        //时间间隔

        )

线程的状态  System.Thread.ThreadState

  

线程上下文获取线程数据 ExecutionContext

异步编程模式APM

为了支持APM模式处理,.NET中对于消耗时间较长的操作,提供了BEGIN和END的方法,当调用Begin,启动异步线程第一步,同时将会保存第三步处理的回调方法,第三部操作将会在线程池的I/O线程上执行。

例子:流的读取操作

1.同步

public override int Read(byte[] array ,int offset, int count);

2.异步,第一步和第三步操作如下

public override IAsyncResult BeginRead(

  byte[] array

  ,int offset

  ,int numBytes

  ,AsyncCallback userCallback

  ,object stateObject)

userCallback:异步调用完成后,系统需要调用的回调方法

stateObject:传递给回调方法的用户自定义的参数对象,一般来说,这个对象是使用异步操作的对象,用来完成结束操作。

public override int EndRead( IAsyncResult syncResult);

FileStream fs= new FileStream(path,open);

IAsyncResult result= fs.BeginRead(buffer,0,4096,delegate(IAsyncResult ar)
{
       int length= fs.EndRead(ar);},fs);

基于事件的异步编程模式EPM

支持EPM的类将有一个或多个后缀为Async的方法,同时还会有相应的名为Completed后缀的事件

Async方法启动异步处理,而Completed事件则通过事件来宣告异步处理的完成

无论处理完成、出现异常、终止异步处理,都必须调用Completed处理程序

例子:WebClient发送一个请求 OpenReadAsync

public void OpenReadAsync(Uri address)

public event OpenReadCompletedEventHandler OpenReadCompleted

处理该事件

public static void OpenReadCallback2(Object sender,OpenReadCompletedEventArgs e)
{
stream reply=null;
streamReader s=null;
try
{ reply=(stream)e.Result;
s=new StreamReader(reply);
Console.WriteLine(s.ReadToEnd());
}

异步线程的状态和同步问题


public interface IAsyncResult

{

  object AsyncState{get;}

  WaitHandle AsyncWatiHanle{get;}

  bool CompletedSynchronously{get;}

  bool IsCompleted{get;}

}

处理管道中的异步问题

将线程从等待中解放出来

在启动线程操作时,将结束异步操作的第三部方法保存起来,在第一步之后,立即结束第一个线程,第二部等待处理完成后,取出保存的第三部处理方法,重新获取一个线程执行第三部,完成后,继续线程的原来的后续步骤。

NextStep类
public NextStep(EndEventHandler handler){
this._endHandler=handler;
}
public void AsyncCallback(IAsyncResult ar)
{ this._endHandler(ar);
}

ReadFile类
封装 IAsyncReault BeginReadFile(Object sender,EvnetArgs e,
   AsyncCallback cb, Object extraData);
       void EndReadFile(IAsyncResult ar);

StepDemo 
封装this._beginHandler和this._endHandler
public void DoAsyncBeginWork()
{
    NextSetp step=new (this._endhandler);
    this.beginHandler(this, , step.AsyncCallback,null);
}

调用
StepDemo demo=new StepDemo();
ReadFile rf=new ReadFile();
demo.AddDoWork(rf.BeginHandler,rf.EndHandler);
demo.DoAsyncBeiginWork();

 线程池

public static class ThreadPool

 .net在每一个线程中维护一个应用程序分享的线程池,线程池提供队列,可以使用少量的线程完成多数线程的任务

线程池工作原理:

  1.线程池的线程分为:计算密集的工作者线程 、 I/O线程

  2.线程池的线程从相应的任务队列中获取相应的任务进行处理,完成后,继续从队列获取任务,如果没有,在等待一段时间后,线程将被销毁

加入工作者线程

  1.public delegate void WaitCallback{object state}

  2.public  static void main(){

    ThreadPool.QueueUserWorkItem(new WaitCallback(ThreadProc));

    static void ThreadProc(object stateInfo){};

加入I/O线程

  调用Begin开头的方法,就是将线程排入线程池的I/O任务队列

HttpApplication中的异步线程

服务器接收一个请求时,httpRuntime将从HttpApplication池中获取一个HttpApplication对象处理请求,请求的处理过程被排入线程池中。

 

推荐阅读