我在一个遗留项目中遇到了一个奇怪的问题,其中事件的组合导致了意外的行为(无论如何我都没有预料到)。我能够在本地复制该问题;并调试以缩小问题范围。我已经在下面详细说明了。谢谢。注意:这是完整的测试复制代码。
开发环境设置
- Windows 7 上的 IIS 7.0
- 集成 AppPool .NET 4.5
- web.config 中的 HttpModule 注册
Http模块
public class Logger : IHttpModule
{
void IHttpModule.Dispose() {}
public void Init(HttpApplication context)
{
context.BeginRequest += new EventHandler(Context_BeginRequest);
}
void Context_BeginRequest(object sender, EventArgs e)
{
var _application = (HttpApplication)sender;
// Comment line below (or change to any other collection) while AutoEventWireUp is true - no problems
foreach (string key in _application.Request.Form.AllKeys) { }
}
}
事件测试.aspx
<%@ Page Language="C#" AutoEventWireup="true" Inherits="EventTest.TestPage" Codebehind="EventTest.aspx.cs" %>
<html xmlns="http://www.w3.org/1999/xhtml">
<head id="Head1" runat="server">
<title>Test</title>
</head>
<body>
<form id="form1" runat="server">
<asp:Label ID="Label1" runat="server"></asp:Label><br />
<asp:Button id="Button1" runat="server" Text="Submit" OnClick="Button1_Click"></asp:Button>
</form>
</body>
</html>
EventTest.aspx.cs
using System;
namespace EventTest
{
public partial class TestPage : System.Web.UI.Page
{
protected void Page_Load(object sender, System.EventArgs e)
{
this.Label1.Text += String.Format("Page_Load fired with PostBack {0}<br />", this.IsPostBack.ToString() );
}
override protected void OnInit(EventArgs e)
{
// Uncomment this and set AutoEventWireUp to False - no problems
//this.Load += new EventHandler(Page_Load);
base.OnInit(e);
}
protected void Button1_Click(object sender, System.EventArgs e)
{
this.Label1.Text += "Button1_Click Fired<br />";
}
}
}
怎么了
如果 AutoEventWireUp 为 True 并且 HttpModule 访问 Request.Form 集合,则不会触发任何事件。
注释 HttpModule 中的 Request.Form 行或将集合切换到 Request.Headers 等。然后所有事件都会触发。
如果 AutoEventWireUp 为 False(并且 Page_Load 是手动注册的),那么无论 Request.Form 访问如何,都会触发所有事件。
这不是我需要为项目解决的问题,因为我不使用 AutoEventWireUp,但我不明白为什么会发生这种情况。如果有人可以对此有所了解;我很感激。
编辑:如果它有帮助,从 Context_PostAcquireRequestState 访问时这不是问题。我很好奇想法,因为 Form 是一个 RO 集合,但似乎有一些修改。