大家好,又见面了,我是你们的朋友全栈君。
Google了一番,大家讨论AutoEventWireup 问题可不少,Page 指令的 AutoEventWireup 属性被设置为 true(或者如果缺少此属性,因为它默认为 true),该页框架将自动调用页事件,即 Page_Init 、 Page_Load等14个方法,在这种情况下,不需要任何显式的 Handles 子句或委托。但这是怎么实现的呢?.net又怎样根据AutoEventWireup 属性来动态编译或者预编译页面呢?我在Google上没有找到答案。
但Scott(K. Scott Allen) 的一篇文章 Inside AutoEventWireup说了当AutoEventWireup为true时,运行时会查找每个方法并注册到相应的事件里,代码就像下面这样:
d = Delegate.CreateDelegate(
typeof(EventHandler), this,
“Page_Load”, ignoreCase,
throwOnFailure
);
Scott说的非常好,但是运行时又是怎么实现的呢?没办法,又是一阵猛翻asp.net源码,终于找到了,代码如下:
// Skip if we’re only generating the intermediate class
if (_sourceDataClass == null)
return;
CodeMemberProperty prop;
// If FAutoEventWireup is turned off, generate a SupportAutoEvents prop that
// returns false.
if (!Parser.FAutoEventWireup)
{
prop = new CodeMemberProperty();
prop.Attributes &= ~MemberAttributes.AccessMask;
prop.Attributes &= ~MemberAttributes.ScopeMask;
prop.Attributes |= MemberAttributes.Override | MemberAttributes.Family;
prop.Name = “SupportAutoEvents”;
prop.Type = new CodeTypeReference(typeof(bool));
prop.GetStatements.Add(new CodeMethodReturnStatement(new CodePrimitiveExpression(false)));
_sourceDataClass.Members.Add(prop);
return;
}
}
BuildAutomaticEventHookup()方法当AutoEventWireup为false时生成了一个重载TemplateControl类的属性SupportAutoEvents,在TemplateControl里该属性返回true,重载代码如下:
而在TemplateControl类里通过调用HookUpAutomaticHandlers()方法来进行自动事件处理函数注册:
if (o == null)
{
dictionary = new ListDictionary();
GetDelegateInformation(dictionary);
if (dictionary.Count == 0)
{
o = _emptyEventSingleton;
}
else
{
o = dictionary;
}
// 缓存
_eventListCache[GetType()] = o;
}
}
}
if (o == _emptyEventSingleton)
{
return;
}
dictionary = (IDictionary)o;
foreach (string key in dictionary.Keys)
{
EventMethodInfo info = (EventMethodInfo)dictionary[key];
Debug.Assert(_eventObjects[key] != null);
bool eventExists = false;
MethodInfo methodInfo = info.MethodInfo;
Delegate eventDelegates = Events[_eventObjects[key]];
if (eventDelegates != null)
{
foreach (Delegate eventDelegate in eventDelegates.GetInvocationList())
{
// 已经添加
if (eventDelegate.Method.Equals(methodInfo))
{
eventExists = true;
break;
}
}
}
if (!eventExists)
{
// 代理委托
IntPtr functionPtr = methodInfo.MethodHandle.GetFunctionPointer();
EventHandler handler = (new CalliEventHandlerDelegateProxy(this, functionPtr, info.IsArgless)).Handler;
Events.AddHandler(_eventObjects[key], handler);
}
}
}
而GetDelegateInformation()方法通过多次调用后会调用到GetDelegateInformationFromMethod()方法来实现根据相应方法(Page_Load等)建立委托,代码如下:
if (e != null)
{
dictionary[methodName] = new EventMethodInfo(e.Method, false);
return true;
}
// 如果不是则找VoidMethod的形式了
VoidMethod vm = (VoidMethod)Delegate.CreateDelegate(typeof(VoidMethod), this,
methodName, true /*ignoreCase*/, false /*throwOnBindFailure*/);
if (vm != null)
{
dictionary[methodName] = new EventMethodInfo(vm.Method, true);
return true;
}
return false;
}
其中参数methodName就是我们通常使用的Page_Load等等方法名了,方法名是一个私有字符串常量,所以就不能在Page页重写了:
说到这里我想再补充下,上面的方法名常量有16个,但是前面我说了只有14个方法,这又是怎么回事呢?
原来Page_Init,Page_Load,Page_DataBind,Page_PreRender,Page_Unload,Page_Error这六个方法无论TemplateControl实例是什么(TemplateControl实例可能是用户控件,或者页面,因为UserControl和Page类都是继承于TemplateControl类的)都会自动注册到相应的事件中去,而Page_PreInit,Page_InitComplete,Page_PreLoad,
Page_LoadComplete,Page_PreRenderComplete,Page_SaveStateComplete只有在TemplateControl实例同样是Page类实例的情况下才会自动注册到相应的事件中去。仔细看剩下的四个方法名,就会发现这是相对应的两组事件处理函数名,默认会添加Page_AbortTransaction和Page_CommitTransaction,如果找不到这两个方法则会查找OnTransactionAbort和OnTransactionCommit方法进行注册,所以说页面会自动注册14个事件处理函数到相应的事件中去(前提是只有这些方法都存在,才会添加成功哦)。
附注:AbortTransaction和CommitTransaction事件用于Enterprise Service Transactions的情况下(@Page指令必须含有Transaction属性,如Transaction=“Required”). 如果需要了解更多信息,可以看看意大利人Alberto Venditti 04年在codeproject上的这篇文章 .NET Distributed Transactions on Enterprise Services: a demo ,或者联系Scott(此Scott仍然是Allen,并非Scott Guthrie),我看过他的博客,他可是这方面的行家。
原文:http://www.0431domain.cn/news/news.aspx/5193
//=================================================
发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/140046.html原文链接:https://javaforall.cn
【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛
【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...