从源码看ASP.NET框架(一)-打造页面控件树
测试实例如下:
前台代码MyFirstWeb.aspx(没有服务器控件,即没有runat)
CodeBehind=”MyFirstWeb.aspx.cs”:表示代码后置类文件 Inherits=”SimpleWeb.MyfirstWeb”:表示该前台类继承于代码后置类文件中的哪个类 |
<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="MyFirstWeb.aspx.cs" Inherits="SimpleWeb.MyFirstWeb" %> <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml">
<head runat="server">
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>
<title></title>
</head>
<body>
<!--获取前台页面的程序集位置-->
<%=this.GetType().Assembly.Location %>
<form id="form1">
<div>
<input type="text" name="txtName"/>
<input type="submit" id="btnClick"/>
</div>
</form>
</body>
</html>
后台代码
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.UI;
using System.Web.UI.WebControls; namespace SimpleWeb
{
public partial class MyFirstWeb : System.Web.UI.Page
{
protected void Page_Load(object sender, EventArgs e)
{
Response.Write("Page_Load方法的开始<br/>");
Response.Write("Page_Load方法的结束<br/>");
}
}
}
一个前台页面文件,在第一次被访问时,会被编译成一个类
前台页面会被编译成前台页面类myfirstweb_aspx,继承于后台页面类MyFirstWeb.cs
发现前台类实现了IRequesSessionState:要取出Session中值,需要改接口 又实现了IHttpHandler:因为后台类继承于Page,Page已经实现了该接口,此处只为方便阅读而已 |
public class myfirstweb_aspx : MyFirstWeb, IRequiresSessionState, IHttpHandler
{
// Fields
private static object __fileDependencies;
private static bool __initialized;
private static MethodInfo __PageInspector_BeginRenderTracingMethod;
private static MethodInfo __PageInspector_EndRenderTracingMethod;
private static MethodInfo __PageInspector_SetTraceDataMethod; // Methods
static myfirstweb_aspx();
[DebuggerNonUserCode]
public myfirstweb_aspx();
[DebuggerNonUserCode]
private HtmlHead __BuildControl__control2();
[DebuggerNonUserCode]
private HtmlMeta __BuildControl__control3();
[DebuggerNonUserCode]
private HtmlTitle __BuildControl__control4();
[DebuggerNonUserCode]
private void __BuildControlTree(myfirstweb_aspx __ctrl);
private void __PageInspector_BeginRenderTracing(object[] parameters);
private void __PageInspector_EndRenderTracing(object[] parameters);
private static MethodInfo __PageInspector_LoadHelper(string helperName);
private void __PageInspector_SetTraceData(object[] parameters);
private void __Render__control1(HtmlTextWriter __w, Control parameterContainer);
[DebuggerNonUserCode]
protected override void FrameworkInitialize();
[DebuggerNonUserCode]
public override int GetTypeHashCode();
[DebuggerNonUserCode]
public override void ProcessRequest(HttpContext context); // Properties
protected HttpApplication ApplicationInstance { get; }
protected DefaultProfile Profile { get; }
}
后台页面类MyFirstWeb.cs,如下
public class MyFirstWeb : Page
{
// Fields
protected HtmlForm form1; // Methods
protected void Page_Load(object sender, EventArgs e)
{
base.Response.Write("Page_Load方法的开始<br/>");
base.Response.Write("Page_Load方法的结束<br/>");
}
}
-------------------------------------------------------------开始分析源码-----------------------------------------------------------------------------------
1.当在浏览框中输入Localhost:MyFirstWeb.aspx,按回车的时候,
在请求管道的第8个事件,创建了与请求名同名的页面类对象myfirstweb_aspx.cs,
然后将该页面类对象保存到HttpContext中的RemapHandler中。
接着在管道的第11个事件和12个事件之间,调用页面对象的PR方法,
因为前台页面类继承于后台类,而后台类有继承于Page类,所以实际调用的是Page类的PR方法。
private void ProcessRequest()
{
Thread currentThread = Thread.CurrentThread;
CultureInfo currentCulture = currentThread.CurrentCulture;
CultureInfo currentUICulture = currentThread.CurrentUICulture;
try
{
this.ProcessRequest(true, true);
}
finally
{
this.RestoreCultures(currentThread, currentCulture, currentUICulture);
}
}
1.1继续调用它的重载函数的PR方法
this.FrameworkInitialize(); 表示调用当前对象的FI方法,当前对象就是前台页面类,那么我们就转到前台页面类的FI方法看看; protected override void FrameworkInitialize(); 发现在前台页面类中,重写了其父类的FI方法 |
private void ProcessRequest(bool includeStagesBeforeAsyncPoint, bool includeStagesAfterAsyncPoint)
{
if (includeStagesBeforeAsyncPoint)
{
this.FrameworkInitialize();
base.ControlState = ControlState.FrameworkInitialized;
}
bool flag = this.Context.WorkerRequest is IIS7WorkerRequest;
try
{
try
{
if (this.IsTransacted)
{
this.ProcessRequestTransacted();
}
else
{
this.ProcessRequestMain(includeStagesBeforeAsyncPoint, includeStagesAfterAsyncPoint);
}
if (includeStagesAfterAsyncPoint)
{
flag = false;
this.ProcessRequestEndTrace();
}
}
catch (ThreadAbortException)
{
try
{
if (flag)
{
this.ProcessRequestEndTrace();
}
}
catch
{
}
}
finally
{
if (includeStagesAfterAsyncPoint)
{
this.ProcessRequestCleanup();
}
}
}
catch
{
throw;
}
}
2.FI方法,代码如下
this.__BuildControlTree(this); 将前台页面对象作为参数传入,开始创建控件树 |
[DebuggerNonUserCode]
protected override void FrameworkInitialize()
{
base.FrameworkInitialize();
this.__BuildControlTree(this);
base.AddWrappedFileDependencies(__fileDependencies);
base.Request.ValidateInput();
}
2.1.__BuildControlTree()代码如下
myfirstweb_aspx __ctrl:那么_ctrl就是前台页面类; IParserAccessor __parser = __ctrl:将_ctrl转为接口类型,那么此时_parser就是前台页面类; __parser.AddParsedSubObject(__ctrl1):将_ctrl1添加到前台页面类中。 问题一:_ctrl1是什么类型?还有它被添加到前台页面类的哪个成员中? |
[DebuggerNonUserCode]
private void __BuildControlTree(myfirstweb_aspx __ctrl)
{
this.InitializeCulture();
HtmlHead __ctrl1 = this.__BuildControl__control2();
IParserAccessor __parser = __ctrl;
__parser.AddParsedSubObject(__ctrl1);
__ctrl.SetRenderMethodDelegate(new RenderMethod(this.__Render__control1));
}
2.1.1.__BuildControl__control2();代码如下
HtmlHead __ctrl = new HtmlHead("head"):创建一个head标签,并添加到前台页面类中; HtmlMeta __ctrl1 = this.__BuildControl__control3():设置meta的属性,并添加到head中; HtmlTitle __ctrl2 = this.__BuildControl__control4():设置标题,并返回标题,然后添加到前台页面对象中; |
[DebuggerNonUserCode]
private HtmlHead __BuildControl__control2()
{
HtmlHead __ctrl = new HtmlHead("head");
HtmlMeta __ctrl1 = this.__BuildControl__control3();
IParserAccessor __parser = __ctrl;
__parser.AddParsedSubObject(__ctrl1);
HtmlTitle __ctrl2 = this.__BuildControl__control4();
__parser.AddParsedSubObject(__ctrl2);
object[] CS$0$0001 = new object[5];
CS$0$0001[0] = __ctrl;
CS$0$0001[2] = 180;
CS$0$0001[3] = 0x79;
CS$0$0001[4] = false;
this.__PageInspector_SetTraceData(CS$0$0001);
return __ctrl;
}
2.1.2.SetRenderMethodDelegate(new RenderMethod());调用一个委托,RenderMethod代码如下
__w.Write:生成的代码写入了一个字符数组中(详情看超链接,文章后续会发布),当输出的时候,也是按写入时的顺序;
__w.Write(base.GetType().Assembly.Location):很熟悉,就是我们在前台中写的C#代码,照样写入到字符数组中; RenderControl(__w):也是吧代码写到字符数组中,不信自己看看源码 |
private void __Render__control1(HtmlTextWriter __w, Control parameterContainer)
{
this.__PageInspector_BeginRenderTracing(new object[] { __w, "/MyFirstWeb.aspx", 0x70, 0x44, true });
__w.Write("\r\n\r\n<!DOCTYPE html>\r\n\r\n<html xmlns=\"http://www.w3.org/1999/xhtml\">\r\n");
this.__PageInspector_EndRenderTracing(new object[] { __w });
parameterContainer.Controls[0].RenderControl(__w);
this.__PageInspector_BeginRenderTracing(new object[] { __w, "/MyFirstWeb.aspx", 0x12d, 0x27, true });
__w.Write("\r\n<body>\r\n <!--获取前台页面的程序集位置-->\r\n ");
this.__PageInspector_EndRenderTracing(new object[] { __w });
this.__PageInspector_BeginRenderTracing(new object[] { __w, "/MyFirstWeb.aspx", 340, 0x26, false });
__w.Write(base.GetType().Assembly.Location);
this.__PageInspector_EndRenderTracing(new object[] { __w });
this.__PageInspector_BeginRenderTracing(new object[] { __w, "/MyFirstWeb.aspx", 0x17a, 0xa7, true });
__w.Write("\r\n <form id=\"form1\" >\r\n <div>\r\n <input type=\"text\" name=\"txtName\"/>\r\n <input type=\"submit\" id=\"btnClick\"/>\r\n </div>\r\n </form>\r\n</body>\r\n</html>\r\n");
this.__PageInspector_EndRenderTracing(new object[] { __w });
}
---------------------------------------------------------------问题解答--------------------------------------------------------------------------------------
问题一:_ctrl1是什么类型?还有它被添加到前台页面类的哪个成员中?
_ctrl1是前台页面类型,那么前台页面又是什么类型
myfirstweb_aspx继承于MyFirstWeb; MyFirstWeb继承于Page; Page继承于TemplateControl;(那么也可以说Page类也是一个控件) TemplateControl继承于Control; Control中有个只读属性,类型是控件集合类型:public virtual ControlCollection Controls { get; } |
那么一切问题都解决了,通过创建控件树,创建了一个个控件,同时将这些控件依次添加到前台页面的控件集合中,然后依次调用控件的RenderControl方法,将结果添加到字符数组中,然后将字符数组输出给HttpRuntime,然后传给iis,最后返回给浏览器,浏览器在根据接收到的结果,渲染界面,这就是我们最终看到的。
源码如下,就是按照字符数组中先后添加的代码显示到界面。为什么Page_Load中的数据先输出,请看超链接,文章后续会发布
Page_Load方法的开始<br/>Page_Load方法的结束<br/> <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml">
<head><meta http-equiv="Content-Type" content="text/html; charset=utf-8" /><title> </title></head>
<body>
<!--获取前台页面的程序集位置-->
D:\Users\Kimisme\AppData\Local\Temp\Temporary ASP.NET Files\root\5f14d39c\6593da16\App_Web_4br2rhb1.dll
<form id="form1" >
<div>
<input type="text" name="txtName"/>
<input type="submit" id="btnClick"/>
</div>
</form> <!-- Visual Studio Browser Link -->
<script type="application/json" id="__browserLink_initializationData">
{"appName":"InternetExplorer","requestId":"6cd255e00ebe439facbd972bb23ed352"}
</script>
<script type="text/javascript" src="http://localhost:1090/86ceaccf0cf9404491a3a03bf9ab677a/browserLink" async="async"></script>
<!-- End Browser Link --> </body>
</html>
从源码看ASP.NET框架(一)-打造页面控件树的更多相关文章
- Asp.net页面生命周期详解任我行(2)-WebForm页面生命周期WEBFORM_ASPNET控件树的生成和作用
摘要 页面类是如何结合后台文件类生成整个页面的HTML的代码和后台输出的代码输出到浏览器中呢?这就牵扯到Asp.net页面生命周期中一个很重要的概念控件树.服务器以反射的方式创建了页面类对象 内容 我 ...
- asp.net原理笔记----页面控件类型,页面状况和asp.net编译过程
通过查看asp.net的整个生命周期之后 了解到在aspx的页面生命周期中 调用了BuildControlTree()方法生成页面控件树 之后再调用Rend()方法根据控件树生成html返回 aspx ...
- Java并发包源码学习之AQS框架(四)AbstractQueuedSynchronizer源码分析
经过前面几篇文章的铺垫,今天我们终于要看看AQS的庐山真面目了,建议第一次看AbstractQueuedSynchronizer 类源码的朋友可以先看下我前面几篇文章: <Java并发包源码学习 ...
- Java并发包源码学习之AQS框架(三)LockSupport和interrupt
接着上一篇文章今天我们来介绍下LockSupport和Java中线程的中断(interrupt). 其实除了LockSupport,Java之初就有Object对象的wait和notify方法可以实现 ...
- Java并发包源码学习之AQS框架(一)概述
AQS其实就是java.util.concurrent.locks.AbstractQueuedSynchronizer这个类. 阅读Java的并发包源码你会发现这个类是整个java.util.con ...
- 从源码看JDK提供的线程池(ThreadPoolExecutor)
一丶什么是线程池 (1)博主在听到线程池三个字的时候第一个想法就是数据库连接池,回忆一下,我们在学JavaWeb的时候怎么理解数据库连接池的,数据库创建连接和关闭连接是一个比较耗费资源的事情,对于那些 ...
- 从微信小程序开发者工具源码看实现原理(一)- - 小程序架构设计
使用微信小程序开发已经很长时间了,对小程序开发已经相当熟练了:但是作为一名对技术有追求的前端开发,仅仅熟练掌握小程序的开发感觉还是不够的,我们应该更进一步的去理解其背后实现的原理以及对应的考量,这可能 ...
- AspNetCore3.1_Secutiry源码解析_8_Authorization_授权框架
目录 AspNetCore3.1_Secutiry源码解析_1_目录 AspNetCore3.1_Secutiry源码解析_2_Authentication_核心流程 AspNetCore3.1_Se ...
- Alink漫谈(二) : 从源码看机器学习平台Alink设计和架构
Alink漫谈(二) : 从源码看机器学习平台Alink设计和架构 目录 Alink漫谈(二) : 从源码看机器学习平台Alink设计和架构 0x00 摘要 0x01 Alink设计原则 0x02 A ...
随机推荐
- 返回通知&异常通知&环绕通知
[返回通知] LoggingAspect.java: @Aspect @Component public class LoggingAspect { /* * 在方法正常执行后执行的通知叫返回通知 * ...
- 【Codeforces 600C】Make Palindrome
[链接] 我是链接,点我呀:) [题意] 题意 [题解] 计算出来每个字母出现的次数. 把字典序大的奇数出现次数的字母换成字典序小的奇数出现次数的字母贪心即可. 注意只有一个字母的情况 然后贪心地把字 ...
- Java基础学习总结(88)——线程创建与终止、互斥、通信、本地变量
线程创建与终止 线程创建 Thread类与 Runnable 接口的关系 public interface Runnable { public abstract void run(); ...
- Uva12657 Boxes in a Line
题目链接:传送门 分析:每次操作都会花费大量时间,显然我们只需要关注每个元素的左边是啥,右边是啥就够了,那么用双向链表,l[i]表示i左边的数,r[i]表示i右边的数,每次操作模拟一下数组的变化就好了 ...
- vue.js通讯----父亲拿儿子的数据
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- NOIP2005 树网的核
题目描述 Description [问题描述]设 T=(V, E, W) 是一个无圈且连通的无向图(也称为无根树),每条边带有正整数的权,我们称T 为树网(treenetwork),其中V, E分别表 ...
- Hihocoder 1329(splay)
Problem 平衡树 Splay 题目大意 维护一个数列,支持三种操作. 操作1:添加一个数x. 操作2:询问不超过x的最大的数. 操作三:删除大小在区间[a,b]内的数. 解题分析 和上一题相比, ...
- [bzoj3668][Noi2014]起床困难综合症_暴力
起床困难综合征 bzoj-3668 Noi-2014 题目大意:题目链接. 注释:略. 想法:Noi考这题...联赛T1难度.... 我们将每个门上的数二进制拆分. 发现:当前位的操作可能直接确定了当 ...
- 洛谷—— P1825 [USACO11OPEN]玉米田迷宫Corn Maze
https://www.luogu.org/problem/show?pid=1825 题目描述 This past fall, Farmer John took the cows to visit ...
- [bzoj 2705][SDOI2012]Longge的问题(数学)
题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=2705 分析: 设k为n的因数 设f[k]为gcd(x,n)==k的x的个数,容易知道a ...