欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

运行page页面时的事件执行顺序及页面的回发与否深度了解

程序员文章站 2024-03-05 16:05:19
复制代码 代码如下: using system; using system.data; using system.configuration; using system.w...
复制代码 代码如下:

using system;
using system.data;
using system.configuration;
using system.web;
using system.web.security;
using system.web.ui;
using system.web.ui.webcontrols;
using system.web.ui.webcontrols.webparts;
using system.web.ui.htmlcontrols;
public partial class _default : page
{
protected void page_load(object sender, eventargs e)
{
}
#region onpreinit 第一步
protected override void onpreinit(eventargs e)
{
//检查 ispostback 属性来确定是不是第一次处理该页。
//创建或重新创建动态控件。
//动态设置主控页。
//动态设置 theme 属性。
//读取或设置配置文件属性值。
//注意
//如果请求是回发请求,则控件的值尚未从视图状态还原。如果在此阶段设置控件属性,则其值可能会在下一事件中被重写。
base.onpreinit(e);
}
#endregion
#region oninit 第二步
protected override void oninit(eventargs e)
{
//在所有控件都已初始化且已应用所有外观设置后引发。使用该事件来读取或初始化控件属性。
base.oninit(e);
}
#endregion
#region oninitcomplete 第三步
protected override void oninitcomplete(eventargs e)
{
//由 page 对象引发。使用该事件来处理要求先完成所有初始化工作的任务。
base.oninitcomplete(e);
}
#endregion
#region preload 第四步
protected override void onpreload(eventargs e)
{
//如果需要在 load 事件之前对页或控件执行处理,请使用该事件。
//在 page 引发该事件后,它会为自身和所有控件加载视图状态,然后会处理 request 实例包括的任何回发数据。
base.onpreload(e);
}
#endregion
#region onload 第五步
protected override void onload(eventargs e)
{
//page 在 page 上调用 onload 事件方法,然后以递归方式对每个子控件执行相同操作,如此循环往复,直到加载完本页和所有控件为止。
//使用 onload 事件方法来设置控件中的属性并建立数据库连接。
base.onload(e);
}
#endregion
#region 控件事件 第六步
protected void button1_click(object sender, eventargs e)
{
//用这些事件来处理特定控件事件,如 button 控件的 click 事件或 textbox 控件的 textchanged 事件。
//注意
//在回发请求中,如果页包含验证程序控件,请在执行任何处理之前检查 page 和各个验证控件的 isvalid 属性。
}
#endregion
#region onloadcomplete 第七步
protected override void onloadcomplete(eventargs e)
{
//对需要加载页上的所有其他控件的任务使用该事件。
base.onloadcomplete(e);
}
#endregion
#region onprerender 第八步
protected override void onprerender(eventargs e)
{
//在该事件发生前:
//page 对象会针对每个控件和页调用 ensurechildcontrols。
//设置了 datasourceid 属性的每个数据绑定控件会调用 databind 方法。有关更多信息,请参见下面的数据绑定控件的数据绑定事件。
//页上的每个控件都会发生 prerender 事件。使用该事件对页或其控件的内容进行最后更改。
base.onprerender(e);
}
#endregion
#region savestatecomplete 第九步
protected override void onsavestatecomplete(eventargs e)
{
//在该事件发生前,已针对页和所有控件保存了 viewstate。将忽略此时对页或控件进行的任何更改。
//使用该事件执行满足以下条件的任务:要求已经保存了视图状态,但未对控件进行任何更改。
base.onsavestatecomplete(e);
}
#endregion
#region render 第十步
//render
//这不是事件;在处理的这个阶段,page 对象会在每个控件上调用此方法。所有 asp.net web 服务器控件都有一个用于写出发送给浏览器的控件标记的 render 方法。
//如果创建自定义控件,通常要重写此方法以输出控件的标记。不过,如果自定义控件只合并标准的 asp.net web 服务器控件,不合并自定义标记,则不需要重写 render 方法。有关更多信息,请参见开发自定义 asp.net 服务器控件。
//用户控件(.ascx 文件)自动合并呈现,因此不需要在代码中显式呈现该控件。
#endregion
#region onunload 第十一步
protected override void onunload(eventargs e)
{
//该事件首先针对每个控件发生,继而针对该页发生。在控件中,使用该事件对特定控件执行最后清理,如关闭控件特定数据库连接。
//对于页自身,使用该事件来执行最后清理工作,如:关闭打开的文件和数据库连接,或完成日志记录或其他请求特定任务。
//注意www.jb51.net
//在卸载阶段,页及其控件已被呈现,因此无法对响应流做进一步更改。如果尝试调用方法(如 response.write 方法),则该页将引发异常。
base.onunload(e);
}
#endregion
}

当页面进行回发时,如点击按钮,以上事件都会重新执行一次,这时的执行顺序为:
1. onpreinit
2. oninit
3. oninitcomplete
4. onpreload
5. page_load
6. onload
7. button_click
8. onloadcomplete
9. onprerender
可以看到,button_click事件位于onload之后执行,可以测试一下:
复制代码 代码如下:

public partial class testcontrols : system.web.ui.page
{
static int count = 0;
protected void page_load(object sender, eventargs e)
{
response.write(count+ "page_load <br />");
count++;
}
protected override void onpreinit(eventargs e)
{
base.onpreinit(e);
response.write(count + "onpreinit <br />");
count++;
}
protected override void oninit(eventargs e)
{
base.oninit(e);
response.write(count + "oninit <br />");
count++;
}
protected override void onload(eventargs e)
{
base.onload(e);
response.write(count + "onload <br />");
count++;
}
protected override void onpreload(eventargs e)
{
base.onpreload(e);
response.write(count + "onpreload <br />");
count++;
}
protected override void onloadcomplete(eventargs e)
{
base.onloadcomplete(e);
response.write(count + "onloadcomplete <br />");
count++;
}
protected override void oninitcomplete(eventargs e)
{
base.oninitcomplete(e);
response.write(count + "oninitcomplete <br />");
count++;
}
protected override void onunload(eventargs e)
{
base.onunload(e);
}
protected override void ondatabinding(eventargs e)
{
base.ondatabinding(e);
response.write(count + "ondatabinding <br />");
count++;
}
protected override void onprerender(eventargs e)
{
base.onprerender(e);
response.write(count + "onprerender <br />");
count++;
}
protected void btngraphics_click(object sender, eventargs e)
{
//bitmap bmp = new bitmap(10, 10);
//graphics g = graphics.fromimage(bmp);
response.write(count + "btngraphics_click <br />");
count++;
}
}

1.熟悉请求管道实现程序运行的全过程:
(1):beginrequest: 开始处理请求
(2):authenticaterequest授权验证请求,获取用户授权信息
(3):postauthenticaterequest获取成功
(4): aunthorizerequest 授权,一般来检查用户是否获得权限
(5):postauthorizerequest:获得授权
(6):resolverequestcache:获取页面缓存结果
(7):postresolverequestcache 已获取缓存
(8):postmaprequesthandler 创建页面对象
(9):acquirerequeststate 获取session-----先判断当前页面对象是否实现了irequiressessionstate接口,如果实现了,则从浏览器发来的请求报文体中获得sessionid,并到服务器的session池中获得对应的session对象,最后赋值给httpcontext的session属性
(10)postacquirerequeststate 获得session
(11)prerequesthandlerexecute:准备执行页面对象
执行页面对象的processrequest方法
(12)postrequesthandlerexecute 执行完页面对象了
(13)releaserequeststate 释放请求状态
(14)postreleaserequeststate 已释放请求状态
(15)updaterequestcache 更新缓存
(16)postupdaterequestcache 已更新缓存
(17)logrequest 日志记录
(18)postlogrequest 已完成日志
(19)endrequest 完成、
复制代码 代码如下:

public class getsession : system.web.ui.page, ireadonlysessionstate
{
string ss = "";
public void init(httpapplication context)
{
//这里可以根据需求,添加各个请求管道
//获取session
context.acquirerequeststate += new eventhandler(context_acquirerequeststate);
//获取url
context.beginrequest += new eventhandler(context_beginrequest);
}
void context_acquirerequeststate(object sender, eventargs e)
{
if (session["user"] != null)
{
ss = session["user"].tostring();
}
}
void context_beginrequest(object sender, eventargs e)
{
//获得当前页面请求管道的httpapplication对象
httpapplication application = sender as httpapplication;
httpcontext context = application.context;//获得上下文对象
string url = context.request.url.localpath;//获得url(不包含域名和路径)
}
}