HttpApplication 异步事件(例如,使用AddOnEndRequestAsync
和 friend 注册的事件)在多大程度上是异步的? IIS 是否等待为特定事件触发的所有异步事件完成,然后再移动到下一个事件,或者它们是否“触发并忘记”?
最佳答案
我不清楚在集成管道模式下运行时的确切工作原理,但我可以告诉你我在非集成情况下看到的情况,并且语义应该保持不变。
简短的回答是每个事件处理程序都是连续触发的,无论是同步的还是异步的,直到前一个事件完成后才会触发下一个事件处理程序。
您可以通过源代码跟踪这一点。
请求进入并存储在队列中。通常,当 HttpRuntime
出队一个请求,它初始化一个 HttpApplication
通过调用其 InitInternal
来获取请求方法,传递 HttpContext
作为论据。HttpApplication.InitInternal
初始化一个新的 HttpApplication.ApplicationStepManager
非集成模式案例的类。然后你可以看到它调用了 BuildSteps
方法就可以了。这将创建一个 ArrayList
存储步骤并构造和存储所有步骤。具体来说,这些步骤是 IExecuteStep
的实现。界面。最终,当所有步骤都添加完毕后,通过将列表复制到数组并将其保存在成员 var _execSteps
中以供以后使用,从而完成列表。 .
这些步骤有多个来源,但您会看到最常用的是HttpApplication.CreateEventExecutionSteps
。它采用事件类型(开始请求、授权等)和步骤数组来添加该事件的步骤。如果您深入了解 CreateEventExecutionSteps
您可以看到它添加了 IExecuteStep
对于它知道的每个异步和同步处理程序,来自 AsyncEvents
和 Events
表,分别。 IExecuteStep
接口(interface)本身基本上由 Execute
组成方法和 CompletedSynchronously
旗帜。
现在,暂停并回顾一下您提到的那些 Add 方法之一,AddOnEndRequestAsync
,您可以看到它将有关异步处理程序的信息添加到 AsyncEvents
table 。 CreateEventExecutionSteps
然后将遍历此表和 AsyncEventExecutionStep
将为添加的每个处理程序构建。
回到请求流程。在 HttpRuntime
之后初始化 HttpApplication
对于请求,它调用它的BeginProcessRequest
方法,触发 ResumeSteps
.ResumeSteps
是重要的一个,您可以在其中看到如何使用这些步骤以及异步情况下的等待策略是什么。你可以看到它维护一个 _currentStepIndex
进入执行步骤数组。最终你会看到它从数组中抓取下一步并调用它的Execute
。方法。如果步骤报告其执行 CompletedSynchronously
,它循环并再次运行。如果没有,它会让方法完成并进入异步深渊。
要查看在这种异步情况下会发生什么,您必须查看 AsyncEventExecutionStep
为异步处理程序创建的实现。在其Execute
实现,您会看到它触发开始处理程序并传入完成回调。在构造函数中,您会看到这个回调初始化为一个最终调用的方法... HttpApplication.ResumeSteps
再次!
因此它继续执行步骤,同步或异步,直到数组溢出,此时它“完成”请求处理。
重点是,您可以清楚地看到,转换为您添加的事件处理程序的步骤是一一执行的,并且无论是同步还是异步,直到当前步骤完成后才会执行以下步骤。您的问题是事件是否以这种方式一一处理,但正如您所见,实际上它更加精细,每个事件处理程序都以这种方式处理,因此每个事件处理程序都可以同步访问 HttpContext 并且可以在没有担心他们是否仍处于管道的“正确阶段”。
显然,源代码中还有其他细节,yada yada,但这是要点。
关于asp.net - 异步 HttpApplication 事件会等到它们返回吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9413585/