自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

ASP.NET Page那點事

開發(fā) 后端
這次的博客我就打算專門談?wù)勊?。不過呢,我不打算說 在Page中使用控件的一些話題,也不會說Page的生命周期的相關(guān)話題,因為我認為這些話題被人談?wù)摰拇螖?shù)實在是太多了,尤其是市面上的ASP.NET的書籍。

Page,我想每個ASP.NET開發(fā)人員對它應(yīng)該都是比較熟悉的。

這次的博客我就打算專門談?wù)勊?。不過呢,我不打算說 在Page中使用控件的一些話題,也不會說Page的生命周期的相關(guān)話題,因為我認為這些話題被人談?wù)摰拇螖?shù)實在是太多了,尤其是市面上的ASP.NET的書籍,都會比較喜歡這些話題。

我不喜歡重復,因此今天我只想談些人家不談的那點事,但我認為它們?nèi)匀缓苤匾?/p>

一些重要的Page指令

雖然Page公開了很多屬性,讓我們可以在運行時調(diào)整它的狀態(tài)與行為,但是,還有些重要的參數(shù)卻是以“指令”方式提供的,需要在設(shè)計時就指定。

下面是我整理的一些我認為 比較重要并且經(jīng)常需要使用的指令:

#p#

web.config的全局設(shè)置

前面我介紹了一些常用的Page指令,考慮到方便性,ASP.NET還允許我們在web.config中為一些常用的指令配置默認值。下面我就一些常用的場景來說明這些全局配置的方便性。

1. 通常,我在創(chuàng)建一個網(wǎng)站項目時,肯定會決定不使用ViewState和Session的。那么如果為每個頁面設(shè)置EnableViewState,EnableSessionState指令屬性,那就顯得太麻煩了,而且還容易遺漏。此時,我們可以直接在web.config中為這些參數(shù)指定一個全局的默認值:

  1. <pages enableViewState="false" enableSessionState="false"></pages> 
  2.  

補充說明一下:全局禁用Session的徹底方法是把Session對應(yīng)的HttpModule從httpModules列表中移除。

web.config允許我們設(shè)置Page默認參數(shù)的具體配置節(jié)如下:

  1. <pages       
  2.    buffer="[True|False]" 
  3.    enableEventValidation="[True|False]" 
  4.    enableSessionState="[True|False|ReadOnly]" 
  5.    enableViewState="[True|False]" 
  6.    enableViewStateMac="[True|False]" 
  7.    smartNavigation="[True|False]" 
  8.    autoEventWireup="[True|False]" 
  9.    pageBaseType="typename, assembly" 
  10.    userControlBaseType="typename" 
  11.    validateRequest="[True|False]" 
  12.    masterPageFile="file path"   
  13.    theme="string" 
  14.    styleSheetTheme="string" 
  15.    maxPageStateFieldLength="number"   
  16.    compilationMode="[Always|Auto|Never]"   
  17.    pageParserFilterType="string"   
  18.    viewStateEncryptionMode="[Always|Auto|Never]"   
  19.    maintainScrollPositionOnPostBack="[True|False]"   
  20.    asyncTimeout="number" 
  21. > 
  22.    <controls>...</controls> 
  23.    <namespaces>...</namespaces> 
  24.    <tagMapping>...</tagMapping> 
  25. </pages> 

2. 為了代碼重用,設(shè)計用戶控件也是很常用的方法。

我們可以使用 @ Register指令 在頁面注冊需要使用的UserControl或者WebControl。然而,有些控件比較通用,許多頁面都會使用它,那么就不要再使用 @ Register指令了,可以在web.config中統(tǒng)一注冊。例如:

  1. <pages> 
  2.     <controls> 
  3.         <add tagPrefix="fish" tagName="MainMenu" src="~/Controls/MainMenu.ascx" /> 
  4.         <add tagPrefix="fish" tagName="PageHeader" src="~/Controls/PageHeader.ascx" /> 
  5.     </controls> 
  6. </pages> 

有了這個定義后,我就可以在任何頁面中直接使用:

  1. <fish:PageHeader runat="server" ID="PageHeader1" /> 

3. 對于喜歡使用頁面內(nèi)聯(lián)代碼的人來說,可能經(jīng)常需要使用自己定義的類型。如果這些類型定義在某個命名空間中,那么就需要在內(nèi)聯(lián)代碼中采用完整命名空間的寫法。雖然這樣做沒有什么問題,但就是麻煩,于是,我們可以在頁面中使用 @ Import指令 來導入我們需要使用的命名空間,但是這個指令每次只能導入一個命名空間,而且每個頁面還得重復導入,顯然不夠方便。

為了方便使用一些常用的命名空間,我們可以在web.config中統(tǒng)一指定,例如:

  1. <pages> 
  2.     <namespaces> 
  3.         <add namespace="MyMVC" /> 
  4.         <add namespace="WebSiteCommonLib" /> 
  5.         <add namespace="WebSiteModel" /> 
  6.     </namespaces> 
  7. </pages> 

這樣設(shè)置后,所有頁面就可以直接使用這些命名空間下的類型了。

不知道有些人想過:為什么在頁面中使用某些微軟提供的類型就不需要導入命名空間?

答案是:其實ASP.NET已經(jīng)將一些微軟認為常用的命名空間在web.config中配置好了:

  1. <pages> 
  2.     <namespaces> 
  3.         <add namespace="System"/> 
  4.         <add namespace="System.Collections"/> 
  5.         <add namespace="System.Collections.Specialized"/> 
  6.         <add namespace="System.Configuration"/> 
  7.         <add namespace="System.Text"/> 
  8.         <add namespace="System.Text.RegularExpressions"/> 
  9.         <add namespace="System.Web"/> 
  10.         <add namespace="System.Web.Caching"/> 
  11.         <add namespace="System.Web.SessionState"/> 
  12.         <add namespace="System.Web.Security"/> 
  13.         <add namespace="System.Web.Profile"/> 
  14.         <add namespace="System.Web.UI"/> 
  15.         <add namespace="System.Web.UI.WebControls"/> 
  16.         <add namespace="System.Web.UI.WebControls.WebParts"/> 
  17.         <add namespace="System.Web.UI.HtmlControls"/> 
  18.     </namespaces> 
  19. </pages> 

4. 現(xiàn)在,有越來越多的人為了方便而使用擴展方法。使用擴展方法的好處是:可以讓我們不去關(guān)心這些擴展方法定義在那個類中,只要在支持擴展方法的對象上調(diào)用就可以了,就像下面的代碼這樣:

  1. 當前用戶已登錄,登錄名:<%= Context.User.Identity.Name.HtmlEncode() %> 

然而,在頁面中使用擴展方法時,也必須先導入擴展方法的定義類的命名空間。

因此,為了方便,我們可以在web.config中為我們定義的擴展方法導入相應(yīng)的命名空間:

  1. <pages> 
  2.     <namespaces> 
  3.         <add namespace="FishDemoCodeLib" /> 
  4.     </namespaces> 
  5. </pages> 

#p#

換個方式使用 Page

在傳統(tǒng)的WEB開發(fā)模式下,我們通常會設(shè)計一些頁面(Page)響應(yīng)來自用戶瀏覽器的請求,在這種模式下,Page會將***生成的整頁HTML代碼直接發(fā)送給用戶瀏覽器。然而,在某些時候,我們只需要生成一個HTML片段:

1. 在AJAX請求中,客戶端為了局部刷新,只要求服務(wù)端返回一個HTML片段。

2. BigPipe方式下,為了能分塊輸出,每次也只需要輸出一個HTML片段。

如果只是為了得到一段簡單的HTML代碼,可能有些人會選擇采用代碼來拼接,但是如果那段HTML還有些復雜呢?顯然拼接方法肯定是不行的。

對于***個問題,可能有人說:我可以創(chuàng)建一個頁面,只放部分代碼到頁面上。的確,這種方法可以勉強解決***個問題,但是,很有可能那部分代碼在整頁輸出時也會用到,怎么辦?

做成UserControl,然后放在一個單獨的頁面中!其實這種做法很無奈,因為那個容器頁面的意義不大(僅僅是個容器),***搞得項目中一大堆頁面文件!事實上,這種方法僅適用于使用簡單服務(wù)端控件的場合,如果想使用一些高級的服務(wù)端控件,它根本就不行。

為了能實現(xiàn)前面說到的二個需求,我們就不能再按照傳統(tǒng)的方式來使用Page了。因為我們希望能得到(返回)一段HTML。

有二種方法可以讓我們繼續(xù)使用頁面模板代碼的方式生成HTML代碼:

1. Server.Execute()方法。

2. Page.RenderControl()方法。

下面這段代碼來源于 MyMVC框架,這個方法可以根據(jù)指定的用戶控件以及控件顯示所需的數(shù)據(jù)得到控件的輸出結(jié)果(一段HTML代碼)。

  1. /// <summary> 
  2. /// 用指定的用戶控件以及視圖數(shù)據(jù)呈現(xiàn)結(jié)果,***返回生成的HTML代碼。  
  3. /// 用戶控件應(yīng)從MyUserControlView<T>繼承  
  4. /// </summary> 
  5. /// <param name="ucVirtualPath">用戶控件的虛擬路徑</param> 
  6. /// <param name="model">視圖數(shù)據(jù)</param> 
  7. /// <returns>生成的HTML代碼</returns> 
  8. public static string Render(string ucVirtualPath, object model)  
  9. {  
  10.     if( string.IsNullOrEmpty(ucVirtualPath) )  
  11.         throw new ArgumentNullException("ucVirtualPath");  
  12.       
  13.     Page page = new Page();  
  14.     Control ctl = page.LoadControl(ucVirtualPath);  
  15.     if( ctl == null )  
  16.         throw new InvalidOperationException(  
  17.             string.Format("指定的用戶控件 {0} 沒有找到。", ucVirtualPath));  
  18.  
  19.     if( model != null ) {  
  20.         MyBaseUserControl myctl = ctl as MyBaseUserControl;  
  21.         if( myctl != null )  
  22.             myctl.SetModel(model);  
  23.     }  
  24.  
  25.     // 將用戶控件放在Page容器中。  
  26.     page.Controls.Add(ctl);  
  27.  
  28.     StringWriter output = new StringWriter();  
  29.     HtmlTextWriter write = new HtmlTextWriter(output, string.Empty);  
  30.     page.RenderControl(write);  
  31.  
  32.     // 用下面的方法也可以的。  
  33.     //HttpContext.Current.Server.Execute(page, output, false);  
  34.  
  35.     return output.ToString();  
  36. }  

整段代碼分為以下幾個步驟(我已用空行分隔開了):

1. 檢查參數(shù)。

2. 創(chuàng)建頁面容器并加載用戶控件。

3. 設(shè)置頁面(視圖)所需的顯示數(shù)據(jù)。

4. 將用戶控件添加到Page的Controls集合中。

5. 調(diào)用RenderControl或者Execute讓Page輸出HTML代碼。

6. 返回結(jié)果。

這段代碼很簡單,唯獨值得介紹的就是第5步,調(diào)用它們就可以得到控件輸出的HTML代碼。

RenderControl或者Execute的差別在于:

RenderControl不支持服務(wù)器控件,原因在于它利用了頁面的一種獨特編譯方式,我已在以前的博客中分析過了。

Execute可以支持服務(wù)器控件,因為它會執(zhí)行一次完整的頁面生命周期。

注意:上面這段代碼就算使用Execute,也只能支持部分簡單的服務(wù)器控件,因為一些復雜的服務(wù)器控件需要在HtmlForm中才能運行。因此,如果需要支持所有的服務(wù)器控件,那么還必須創(chuàng)建HtmlForm對象,并調(diào)整包含關(guān)系,還有就是還需要去掉產(chǎn)生的多余HTML代碼。

如果你需要生成整個頁面生成的HTML代碼,可以參考 MyMVC框架,那里有實現(xiàn)這個功能的完整代碼。

#p#

重新認識Eval()方法

我想很多人都寫過類似下面的代碼:

  1. <asp:Repeater ID="repeater1" runat="server"> 
  2.     <HeaderTemplate><ul></HeaderTemplate> 
  3.     <FooterTemplate></ul></FooterTemplate> 
  4.     <ItemTemplate> 
  5.         <li><%# Eval("OrderID")%><%# Eval("OrderDate")%>,<%# Eval("SumMoney")%> 
  6.         </li> 
  7.     </ItemTemplate> 
  8. </asp:Repeater> 

在這里我要說的是 Eval() 的調(diào)用,還不是Repeater控件。

Eval()不僅僅可以讀取一個綁定數(shù)據(jù)項的屬性,還可以去讀取DataTable中的一個數(shù)據(jù)列。而且還能完成更復雜的綁定計算:

  1. <li><%# Eval("OrderID")%>,<%# Eval("OrderDate")%>,<%# Eval("SumMoney")%> 
  2.     ,訂單中的***個商品:<%# Eval("Detail[0].ProductName") %> 
  3. </li> 

當然了,對于頁面上的數(shù)據(jù)綁定,用Eval()的確不是性能***的方法,建議還是使用強類型轉(zhuǎn)換的方法。

有時候,尤其是在寫反射應(yīng)用時,時常會有從字符串解析并實現(xiàn)求值計算的需求。那么,前面這個示例中,Eval()的功能是不是值得挖掘呢?我認為答案是肯定的。

通過分析ASP.NET的綁定代碼,我發(fā)現(xiàn)Eval在內(nèi)部會調(diào)用DataBinder.Eval這個靜態(tài)方法,這個方法的簽名如下:

  1. //     在運行時計算數(shù)據(jù)綁定表達式。  
  2. //  
  3. // 參數(shù):  
  4. //   container:  
  5. //     表達式根據(jù)其進行計算的對象引用。此標識符必須是以頁的指定語言表示的有效對象標識符。  
  6. //  
  7. //   expression:  
  8. //     從 container 到要放置在綁定控件屬性中的公共屬性值的導航路徑。  
  9. //     此路徑必須是以點分隔的屬性或字段名稱字符串,如 C# 中的 "Tables[0].DefaultView.[0].Price"  
  10. //     或 Visual Basic 中的 "Tables(0).DefaultView.(0).Price"。  
  11. //  
  12. // 返回結(jié)果:  
  13. //     System.Object,它是數(shù)據(jù)綁定表達式的計算結(jié)果。  
  14. public static object Eval(object container, string expression);  

通過這個簽名的注釋,我們可以很容易地看出它的用法。

下面我來舉個例子把它應(yīng)用在非綁定的應(yīng)用中:

我有一個類:

  1. public class TestEvalClass  
  2. {  
  3.     public List<Order> Orders { get; set; }  
  4.  
  5.     // Order以及OrderDetail的定義就省略了,我想大家能想像得出來。  
  6. }  

那么下面的代碼是可以運行的:

  1. static void Main()  
  2. {  
  3.     TestEvalClass testObject = GetTestEvalClassInstance();  
  4.  
  5.     string productName = (string)System.Web.UI.DataBinder.Eval(testObject, "Orders[0].Detail[0].ProductName");  
  6.     Console.WriteLine(productName);  
  7. }  

對于這個示例,我想輸出什么結(jié)果,并不重要。

我只想說:如果讓你去解析那個表達式,會不會比較麻煩,現(xiàn)在有現(xiàn)成的,用起來是不是很方便?

#p#

不用基類也能擴展

在一個ASP.NET網(wǎng)站中,如果想為所有的頁面添加某個功能,我們通常會想到使用基類的方式去實現(xiàn)。這的確是一種很有效的方法,但不并唯一的方法,還有一種方法也能容易實現(xiàn)這個需求,那就是使用PageAdapter的方式。

在我寫博客的過程中,我寫了很多示例頁面,頁面中包含一些提交按鈕是少不了的事情,然而,為了能讓示例代碼看起來比較原始(簡單),我盡量不使用服務(wù)器控件,因此就要面臨提交按鈕的事件處理問題。在博客【細說 ASP.NET Cache 及其高級用法】的示例代碼中,我開始采用PageAdapter這種方法,它可以讓代碼很簡單,而且以后也方便以后重用(只需要復制幾個文件即可)。

或許有些人認為:擴展所有頁面的功能,還是使用基類比較好。

對于這個觀點,我完全不反對。

但是,PageAdapter的好處在于它的可插拔性(類似HttpModule的優(yōu)點)。不過,我當時設(shè)計這種擴展方式只是想再換個方法嘗試一下而已。

其實微軟設(shè)計PageAdapter的本意是為了處理各種瀏覽器的兼容問題,但是我把這個功能用到擴展Page的功能上去了。 HttpModule可以進入到ASP.NET請求管線的任何階段,但它就是進入不了頁面的生命周期中,有了這個方法,我們就可以采用HttpModule這種【外掛】式的方法進入到頁面生命周期中,我認為是很有意義的。

方法多了,我想不是件壞事。每種方法都有適合它們的應(yīng)用場合,了解更多的方法,以后就能做出更優(yōu)秀的設(shè)計。

這次想到這個話題是因為前面的博客【細說ASP.NET Forms 身份認證】中的示例代碼。有些人看到那些代碼,發(fā)現(xiàn)代碼的運行方式比較特別,所以,今天我就打算著重介紹這種方法。

我們再來回顧一下以前博客中的示例代碼,首先從頁面代碼開始:

  1. <fieldset><legend>普通登錄</legend><form action="<%= Request.RawUrl %>" method="post"> 
  2.     登錄名:<input type="text" name="loginName" style="width: 200px" value="Fish" /> 
  3.     <input type="submit" name="NormalLogin" value="登錄" /> 
  4. </form></fieldset> 
  5.    
  6. <fieldset><legend>包含【用戶信息】的自定義登錄</legend>    <form action="<%= Request.RawUrl %>" method="post"> 
  7.     <table border="0"> 
  8.     <tr><td>登錄名:</td> 
  9.         <td><input type="text" name="loginName" style="width: 200px" value="Fish" /></td></tr> 
  10.     <tr><td>UserId:</td> 
  11.         <td><input type="text" name="UserId" style="width: 200px" value="78" /></td></tr> 
  12.     <tr><td>GroupId:</td> 
  13.         <td><input type="text" name="GroupId" style="width: 200px" /> 
  14.         1表示管理員用戶  
  15.         </td></tr> 
  16.     <tr><td>用戶全名:</td> 
  17.         <td><input type="text" name="UserName" style="width: 200px" value="Fish Li" /></td></tr> 
  18.     </table>      
  19.     <input type="submit" name="CustomizeLogin" value="登錄" /> 
  20. </form></fieldset> 

在這段頁面代碼中,我定義了二個表單,它們包含各自的提交按鈕(其實這也只是部分代碼)。

再來看后臺處理代碼是如何響應(yīng)提交請求的:

  1. public partial class _Default : System.Web.UI.Page   
  2. {  
  3.     [SubmitMethod(AutoRedirect = true)]  
  4.     public void NormalLogin()  
  5.     {  
  6.         // 省略登錄處理代碼。  
  7.         // 如果需要知道這段代碼可以瀏覽下面的網(wǎng)址:  
  8.         // http://www.cnblogs.com/fish-li/archive/2012/04/15/2450571.html   
  9.     }  
  10.  
  11.     [SubmitMethod(AutoRedirect = true)]  
  12.     public void CustomizeLogin()  
  13.     {  
  14.         // 省略登錄處理代碼。  
  15.         // 如果需要知道這段代碼可以瀏覽下面的網(wǎng)址:  
  16.         // http://www.cnblogs.com/fish-li/archive/2012/04/15/2450571.html   
  17.     }  

注意觀察,這二個C#方法的名稱與頁面二個submit按鈕的name屬性相同,因此可以猜測到這二個C#方法可以處理那二個submit按鈕的提交請求。那么這二段代碼是如何運行起來的呢?有些人或許看到了[SubmitMethod]的使用,認為與它們有關(guān)。其實這種說法并不正確,我也可以完全不使用它們。請記?。篈ttribute永遠只是一個標記,它不可能讓代碼自動運行起來。

前面的代碼能運行起來,與App_Browsers目錄下的Page.browser文件有關(guān),此文件的代碼如下:

  1. <browsers> 
  2.     <browser refID="Default"> 
  3.         <controlAdapters> 
  4.             <adapter controlType="System.Web.UI.Page" 
  5.                      adapterType="FishDemoCodeLib.MyPageAdapter, FishDemoCodeLib" /> 
  6.         </controlAdapters> 
  7.     </browser> 
  8. </browsers> 

這里定義了一個MyPageAdapter,它用于Page控件的請求過程。 refID="Default" 表示是對ASP.NET定義的Default.browser文件補充一些配置,它將能匹配來自所有瀏覽器的請求。

我再來看一下MyPageAdapter的代碼:

  1. [AttributeUsage(AttributeTargets.Method, AllowMultiple = false)]  
  2. public class SubmitMethodAttribute : Attribute  
  3. {  
  4.     public bool AutoRedirect { get; set; }  
  5. }  
  6.  
  7. internal sealed class MethodInvokeInfo  
  8. {  
  9.     public MethodInfo MethodInfo;  
  10.     public SubmitMethodAttribute MethodAttribute;  
  11. }  
  12.  
  13. public class MyPageAdapter : System.Web.UI.Adapters.PageAdapter  
  14. {  
  15.     private static readonly Hashtable s_table = Hashtable.Synchronized(new Hashtable());  
  16.  
  17.     private static MethodInvokeInfo[] GetMethodInfo(Type type)  
  18.     {  
  19.         MethodInvokeInfo[] array = s_table[type.AssemblyQualifiedName] as MethodInvokeInfo[];  
  20.         if( array == null ) {  
  21.             array = (from m in type.GetMethods(BindingFlags.Instance | BindingFlags.Public)  
  22.                      let a = m.GetCustomAttributes(  
  23.                             typeof(SubmitMethodAttribute), false) as SubmitMethodAttribute[]  
  24.                      where a.Length > 0  
  25.                      select new MethodInvokeInfo {   
  26.                             MethodInfo = mMethodAttribute = a[0] }).ToArray();  
  27.  
  28.             s_table[type.ToString()] = array;  
  29.         }  
  30.         return array;  
  31.     }  
  32.  
  33.  
  34.     protected override void OnLoad(EventArgs e)  
  35.     {  
  36.         base.OnLoad(e);  
  37.  
  38.         if( Page.Request.Form.AllKeys.Length == 0 )  
  39.             return;    // 沒有提交表單  
  40.  
  41.         MethodInvokeInfo[] array = GetMethodInfo(Page.GetType().BaseType);  
  42.         if( array.Length == 0 )  
  43.             return;  
  44.  
  45.         foreach( MethodInvokeInfo m in array ) {  
  46.             if( string.IsNullOrEmpty(Page.Request.Form[m.MethodInfo.Name]) == false ) {  
  47.                 m.MethodInfo.Invoke(Page, null);  
  48.  
  49.                 if( m.MethodAttribute.AutoRedirect   
  50.                                 && Page.Response.IsRequestBeingRedirected == false )  
  51.                     Page.Response.Redirect(Page.Request.RawUrl);  
  52.  
  53.                 return;  
  54.             }  
  55.         }  
  56.     }  
  57. }  

這段代碼并不長,核心代碼更是比較少。

代碼中,最重要的一塊是MyPageAdapter的實現(xiàn),它繼承了System.Web.UI.Adapters.PageAdapter,并重寫了OnLoad方法(相當是在重寫Page的OnLoad方法),也正是由于這個重寫,代碼才有機會在頁面的生命周期中被執(zhí)行,這一點是HttpModule做不到的。

在OnLoad方法中做了以下事情:

1. 檢查是不是發(fā)生了表單提交的操作。

2. 獲取當前頁面類型的所有[SubmitMethod]修飾過的方法。

3. 檢查提交的表單數(shù)據(jù)中,是否存在與name對應(yīng)的C#方法名。

4. 如果找到一個匹配的方法名,則調(diào)用。

5. 如果在[SubmitMethod]中設(shè)置了AutoRedirect=true,則引發(fā)重定向。

注意:如果不調(diào)用base.OnLoad(e); 那么頁面的Load事件根本不會發(fā)生。也就是說:PageAdapter.OnLoad的調(diào)用時間要早于Page.Onload方法。

由于這段代碼僅供我寫示例代碼時使用,因此并沒有檢查要調(diào)用的方法的參數(shù)是否滿足條件,也沒有優(yōu)化刻意去優(yōu)化它的性能。在我的設(shè)計中,被調(diào)用的方法應(yīng)該是無參的,因此是容易判斷的,而且可以使用一個固定簽名的委托去優(yōu)化它的,這些細節(jié)留著以后再去完善它吧!

 

原文鏈接:http://www.cnblogs.com/fish-li/archive/2012/06/10/2544087.html

責任編輯:張偉 來源: Fish Li的博客
相關(guān)推薦

2009-08-07 16:45:09

ASP.NET頁面對象

2009-08-03 14:22:33

什么是ASP.NET

2009-07-28 17:17:19

ASP.NET概述

2009-08-04 17:49:31

Web Page生命周ASP.NET Pos

2009-07-23 14:47:03

Page_Load方法ASP.NET

2009-07-22 17:45:35

ASP.NET教程

2009-07-27 12:22:03

ASP.NET和ASPASP.NET入門教程

2010-05-06 17:46:47

2009-08-10 13:32:15

ASP.NET TimASP.NET組件設(shè)計

2009-07-29 17:11:25

ASP.NET ISA

2021-04-26 14:25:39

ASP.NET Cor單元測試

2009-07-29 16:08:07

ASP和ASP.NET

2009-08-03 17:35:07

ASP.NET WebASP.NET編程工具

2009-07-28 09:02:32

asp.net aja

2009-08-03 13:38:18

ASP.NET編程模型

2009-07-27 14:46:16

XML和ASP.NET

2009-07-23 13:19:51

2009-08-19 09:23:40

ASP.NET Rou

2009-07-31 10:23:44

緩存頁面ASP.NET緩存

2015-06-11 14:57:23

ASP.NET 5
點贊
收藏

51CTO技術(shù)棧公眾號