Global.asax 和 MasterPage 之间发生了什么?

What happens between Global.asax and MasterPage?

我正在开始多语言工作。在 Global.asax.cs 方法 Application_BeginRequest() 中,读取一个 cookie 并将 CurrentUICulture 设置为 en-US。但是,在 MyPage.aspx 中,值意外地变为 nl

现在这个网站是我继承的,中等规模,有一个相当复杂的菜单系统。此外,它维护了一个用户 table,其中有一个首选语言字段,其值如 nl,但我(还)找不到从该用户 table 设置 CurrentUICulture 的代码。

这是一个有MasterPage的页面,我就往那边看了。我在 MasterPage.Page_Init() 的第一个页面事件中设置了一个调试断点,并在立即 Window 中检查了 System.Threading.Thread.CurrentThread.CurrentUICulture.Name。值为:'nl'。我完全不解。

我的问题是:在 Global.asax.cs、Application_BeginRequest() 和 MasterPage.Page_Init() 之间可能执行什么代码?

Application_BeginRequest() 处理程序是 Asp.Net 生命周期的第一步。看这里:

https://docs.microsoft.com/en-us/iis/application-frameworks/building-and-running-aspnet-applications/aspnet-integration-with-iis

当处理程序正在执行时(即您的页面),它有自己的生命周期:

https://docs.microsoft.com/en-us/previous-versions/ms178472(v=vs.140)?redirectedfrom=MSDN

并且该文档说:

Start

In the start stage, page properties such as Request and Response are set. At this stage, the page also determines whether the request is a postback or a new request and sets the IsPostBack property. The page also sets the UICulture property.

选中可用事件后,我们看到:

PreInit

Raised after the start stage is complete and before the initialization stage begins.

因此;似乎您可以更改 UICulture 的最早阶段是 PreInit 阶段。然而,最合适的地方是服务于此特定目的的 InitializeCulture 方法:

public partial class _Default : Page
{
    protected override void OnPreInit(EventArgs e)
    {
        base.OnPreInit(e);

        HttpCookie languageCookie = Request.Cookies["lang"];
        if (languageCookie != null)
        {
            if (languageCookie.Value == "en")
            {
                base.Culture = base.UICulture = CultureInfo.GetCultureInfo("en-US").Name;
            }
        }
    }

    protected override void InitializeCulture()
    {
        // Or do it here. This is more appropriate.
    }
    ...
    ...
}

编辑:

尽管跨多个页面共享文化信息,但不应在母版页的事件处理程序中设置文化信息,因为从以下跟踪输出中可以看出,当页面使用母版页时,母版页将被解释作为页面的控件,此母版页的 Page_Load 事件在 LoadControls() 期间执行,该事件在页面的 Page_Load 处理程序之后执行。

我还想在这里提到跟踪功能。通过启用跟踪,可以看到很多关于页面执行过程的信息,它的children,所有的时间,headers等等:

在 web.config 文件中,在 <system.web> 下:

<trace pageOutput="true" requestLimit="10" enabled="true" localOnly="true" traceMode="SortByTime" mostRecent="true"/>

或在此处阅读 <system.webServer> 等价物:

https://docs.microsoft.com/en-us/iis/configuration/system.webserver/tracing/

Control Tree
Control UniqueID    Type    Render Size Bytes (including children)  ViewState Size Bytes (excluding children)   ControlState Size Bytes (excluding children)
__Page  ASP.default_aspx    1104    0   0
    ctl00   ASP.masterpage_master   1104    0   0
        ctl00$ctl02 System.Web.UI.LiteralControl    68  0   0
        ctl00$ctl00 System.Web.UI.HtmlControls.HtmlHead 48  0   0
            ctl00$ctl01 System.Web.UI.HtmlControls.HtmlTitle    29  0   0
            ctl00$head  System.Web.UI.WebControls.ContentPlaceHolder    6   0   0
                ctl00$head$ctl00    System.Web.UI.LiteralControl    6   0   0
        ctl00$ctl03 System.Web.UI.LiteralControl    14  0   0
        form1   System.Web.UI.HtmlControls.HtmlForm 954 0   0
            ctl00$ctl04 System.Web.UI.LiteralControl    21  0   0
            ctl00$ContentPlaceHolder1   System.Web.UI.WebControls.ContentPlaceHolder    291 0   0
                ctl00$ContentPlaceHolder1$ctl00 System.Web.UI.LiteralControl    198 0   0
                ctl00$ContentPlaceHolder1$hdn1  System.Web.UI.WebControls.HiddenField   91  0   0
                ctl00$ContentPlaceHolder1$ctl01 System.Web.UI.LiteralControl    2   0   0
            ctl00$ctl05 System.Web.UI.LiteralControl    18  0   0
        ctl00$ctl06 System.Web.UI.LiteralControl    20  0   0