溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

編寫輕量ajax組件01-與webform平臺上的各種實現(xiàn)方式比較的示例分析

發(fā)布時間:2021-09-17 16:18:11 來源:億速云 閱讀:111 作者:柒染 欄目:web開發(fā)

編寫輕量ajax組件01-與webform平臺上的各種實現(xiàn)方式比較的示例分析,針對這個問題,這篇文章詳細介紹了相對應的分析和解答,希望可以幫助更多想解決這個問題的小伙伴找到更簡單易行的方法。

前言

  Asp.net WebForm 和 Asp.net MVC(簡稱MVC) 都是基于Asp.net的web開發(fā)框架,兩者有很大的區(qū)別,其中一個就是MVC更加注重http本質(zhì),而WebForm試圖屏蔽http,為此提供了大量的服務器控件和ViewState機制,讓開發(fā)人員可以像開發(fā)Windows Form應用程序一樣,基于事件模型去編程。兩者各有優(yōu)缺點和適用情景,但MVC現(xiàn)在是許多Asp.net開發(fā)者的首選。

  WebForm是建立在Asp.net的基礎上的,Asp.net提供了足夠的擴展性,我們也可以利用這些在WebForm下編寫像MVC一樣的框架,這個有機會再寫。說到WebForm很多人就會聯(lián)想到服務器控件(拖控件?。。。?,其實不然,我們也可以完全不使用服務器控件,像MVC那樣關注html。WebForm要拋棄服務器控件,集中關注html,首先就要將<form runat="server"></form>標簽去掉,這個runat server 的form 是其PostBack機制的基礎。既然我們要回歸到html+css+js,那么意味著許多東西都要自己實現(xiàn),例如處理Ajax請求。不像MVC那樣,WebForm開始的設計就將服務器控件作為主要組成部分,如果不使用它,那么只能利用它的擴展性去實現(xiàn)。

本系列就是實現(xiàn)一個基于WebForm平臺的輕量級ajax組件,主要分為三個部分:

  1. 介紹WebForm下各種實現(xiàn)方式。

  2. 分析ajaxpro組件。

  3. 編寫自己的ajax組件。

一、Ajax簡介

  異步允許我們在不刷新整個頁面的情況下,像服務器請求或提交數(shù)據(jù)。對于復雜的頁面,為了請求一點數(shù)據(jù)而重載整個頁面顯然是很低效的,ajax就是為了解決這個問題的。ajax的核心是XmlHttpRequest對象,通過該對象,以文本的形式向服務器提交請求。XmlHttpRequest2.0后,還支持提交二進制數(shù)據(jù)。

  ajax安全:出于安全考慮,ajax受同源策略限制;也就是只能訪問同一個域、同一個端口的請求,跨域請求會被拒絕。當然有時候需求需要跨域發(fā)送請求,常用的跨域處理方法有CORS(跨域資源共享)和JSONP(參數(shù)式JSON)。

  ajax數(shù)據(jù)交互格式:雖然Ajax核心對象XmlHttpRequest有"XML"字眼,但客戶端與服務器數(shù)據(jù)交換格式不局限于xml,例如現(xiàn)在更多是使用json格式?! ?/p>

  ajax 也是有缺點的。例如對搜索引擎的支持不太好;有時候也會違背url資源定位的初衷。

二、Asp.net MVC 平臺下使用ajax

  在MVC里,ajax調(diào)用后臺方法非常方便,只需要指定Action的名稱即可。

  前臺代碼:

<body>
  <h2>index</h2>
  <input type="button" value="GetData" onclick="getData()" />
  <span id="result"></span>
</body>
<script type="text/javascript">
  function getData() {
    $.get("GetData", function (data) {
      $("#result").text(data);
    });
  }
</script>

  后臺代碼:

public class AjaxController : Controller
{
  public ActionResult GetData()
  {
    if(Request.IsAjaxRequest())
    {
      return Content("data");
    }
    return View();
  }
}

三、WebForm 平臺下使用ajax

  3.1 基于服務器控件包或者第三方組件

  這是基于服務器控件的,例如ajax toolkit工具包,或者像FineUI這樣的組件。web前端始終是由html+css+js組成的,只不過如何去生成的問題。原生的我們可以自己編寫,或者用一些前端插件;基于服務器控件的,都是在后臺生成的,通常效率也低一點。服務器組件會在前臺生成一系列代理,本質(zhì)還是一樣的,只不過控件封裝了這個過程,不需要我們自己編寫?;诳丶蛘叩谌浇M件的模式,在一些管理系統(tǒng)還是挺有用的,訪問量不是很大,可以快速開發(fā)。

  3.2 基于ICallbackEventHandler接口

  .net 提供了ICallbackEventHandler接口,用于處理回調(diào)請求。該接口需要用ClientScriptManager在前臺生成代理腳本,用于發(fā)送和接收請求,所以需要<form runat="server">標簽。

  前臺代碼:

<body>
  <form id="form1" runat="server">
  <div>    
    <input type="button" value="獲取回調(diào)結果" onclick="callServer()" />
    <span id="result" ></span>
  </div>
  </form>
</body>
<script type="text/javascript">
  function getCallbackResult(result){
    document.getElementById("result").innerHTML = result;
  }
</script>

  后臺代碼:

public partial class Test1 : System.Web.UI.Page, ICallbackEventHandler
{    
  protected void Page_Load(object sender, EventArgs e)
  {
    //客戶端腳本Manager
    ClientScriptManager scriptMgr = this.ClientScript;
 
    //獲取回調(diào)函數(shù),getCallbackResult就是回調(diào)函數(shù)
    string functionName = scriptMgr.GetCallbackEventReference(this, "", "getCallbackResult", "");
 
    //發(fā)起請求的腳本,callServer就是點擊按鈕事件的執(zhí)行函數(shù)
    string scriptExecutor = "function callServer(){" + functionName + ";}";
 
    //注冊腳本
    scriptMgr.RegisterClientScriptBlock(this.GetType(), "callServer", scriptExecutor, true);
  }
 
  //接口方法
  public string GetCallbackResult()
  {
    return "callback result";
  }
 
  //接口方法
  public void RaiseCallbackEvent(string eventArgument)
  {
  }
}

  這種方式有以下缺點

  1. 實現(xiàn)起來較復雜,每個頁面Load事件都要去注冊相應的腳本。

  2. 前臺會生成一個用于代理的腳本文件。

  3. 對于頁面交互復雜的,實現(xiàn)起來非常麻煩。

  4. 雖然是回調(diào),但是此時頁面對象還是生成了。

  3.3 使用一般處理程序

  一般處理程序其實是一個實現(xiàn)了IHttpHandler接口類,與頁面類一樣,它也可以用于處理請求。一般處理程序通常不用于生成html,也沒有復雜的事件機制,只有一個ProcessRequest入口用于處理請求。我們可以將ajax請求地址寫成.ashx文件的路徑,這樣就可以處理了,而且效率比較高。

  要輸出文本內(nèi)容只需要Response.Write(data)即可,例如,從數(shù)據(jù)庫獲取數(shù)據(jù)后,序列化為json格式字符串,然后輸出。前面說到,一般處理程序不像頁面一樣原來生成html,如果要生成html,可以通過加載用戶控件生成。如:

public void ProcessRequest(HttpContext context)
{
  Page page = new Page();
  Control control = page.LoadControl("~/PageOrAshx/UserInfo.ascx");
  if (control != null)
  {
    StringWriter sw = new StringWriter();
    HtmlTextWriter writer = new HtmlTextWriter(sw);
    control.RenderControl(writer);
    string html = sw.ToString();
    context.Response.Write(html);        
  }
}

  這種方式的優(yōu)點是輕量、高效;缺點是對于交互多的需要定義許多ashx文件,加大了管理和維護成本。

  3.4 頁面基類

  將處理ajax請求的方法定義在頁面對象內(nèi),這樣每個頁面就可以專注處理本頁面相關的請求了。這里有點需要注意。

  1.如何知道這個請求是ajax請求?

    通過請求X-Requested-With:XMLHttlRequest 可以判斷,大部份瀏覽器的異步請求都會包含這個請求頭;也可以通過自定義請求頭實現(xiàn),例如:AjaxFlag:XHR。

  2.在哪里統(tǒng)一處理?

    如果在每個頁面類里判斷和調(diào)用是很麻煩的,所以將這個處理過程轉到一個頁面基類里處理。

  3.如何知道調(diào)用的是哪個方法?

    通過傳參或者定義在請求頭都可以,例如:MethodName:GetData。

  4.知道方法名稱了,如何動態(tài)調(diào)用?

    反射。

  5.如何知道該方法可以被外部調(diào)用?

    可以認為public類型的就可以被外部調(diào)用,也可以通過標記屬性標記。

  通過上面的分析,簡單實現(xiàn)如下  

  頁面基類:

public class PageBase : Page
{
  public override void ProcessRequest(HttpContext context)
  {
    HttpRequest request = context.Request;
    if (string.Compare(request.Headers["AjaxFlag"],"AjaxFlag",0) == 0)
    {
      string methodName = request.Headers["MethodName"];
      if (string.IsNullOrEmpty(methodName))
      {
        EndRequest("MethodName標記不能為空!");
      }
      Type type = this.GetType().BaseType;
      MethodInfo info = type.GetMethod(methodName, BindingFlags.Public | BindingFlags.Instance | BindingFlags.Static);
      if (info == null)
      {
        EndRequest("找不到合適的方法調(diào)用!");
      }        
      string data = info.Invoke(this, null) as string;
      EndRequest(data);
    }
    base.ProcessRequest(context);
  }
  private void EndRequest(string msg)
  {
    HttpResponse response = this.Context.Response;
    response.Write(msg);
    response.End();
  }
}

  頁面類:

public partial class Test1 : PageBase
{
  protected void Page_Load(object sender, EventArgs e)
  {
  }
  public string GetData()
  {
    return "213";
  }
}

  前臺代碼:

function getData(){
  $.ajax({
    headers:{"AjaxFlag":"XHR","MethodName":"GetData"},
    success:function(data){
      $("#result").text(data);
    }
  });
}

四、優(yōu)化版頁面基類

  上面的頁面基類功能很少,而且通過反射這樣調(diào)用的效率很低。這里優(yōu)化一下:

  1.可以支持簡單類型的參數(shù)。

    例如上面的GetData可以是:GetData(string name),通過函數(shù)元數(shù)據(jù)可以獲取相關的參數(shù),再根據(jù)請求的參數(shù),就可以設置參數(shù)了。

  2.加入標記屬性。

    只有被AjaxMethodAttribute標記的屬性才能被外部調(diào)用。

  3.優(yōu)化反射。

    利用緩存,避免每次都根據(jù)函數(shù)名稱去搜索函數(shù)信息。

  標記屬性:

public class AjaxMethodAttribute : Attribute
{
}

  緩存對象:  

public class CacheMethodInfo
{
  public string MethodName { get; set; }
  public MethodInfo MethodInfo { get; set; }
  public ParameterInfo[] Parameters { get; set; }
}

  基類代碼:

public class PageBase : Page
{
  private static Hashtable _ajaxTable = Hashtable.Synchronized(new Hashtable());
  public override void ProcessRequest(HttpContext context)
  {      
    HttpRequest request = context.Request;
    if (string.Compare(request.Headers["AjaxFlag"],"XHR",true) == 0)
    {
      InvokeMethod(request.Headers["MethodName"]);
    }
    base.ProcessRequest(context);
  }
  /// <summary>
  /// 反射執(zhí)行函數(shù)
  /// </summary>
  /// <param name="methodName"></param>
  private void InvokeMethod(string methodName)
  {
    if (string.IsNullOrEmpty(methodName))
    {
      EndRequest("MethodName標記不能為空!");
    }
    CacheMethodInfo targetInfo = TryGetMethodInfo(methodName);
    if (targetInfo == null)
    {
      EndRequest("找不到合適的方法調(diào)用!");
    }
    try
    {
      object[] parameters = GetParameters(targetInfo.Parameters);
      string data = targetInfo.MethodInfo.Invoke(this, parameters) as string;
      EndRequest(data);
    }
    catch (FormatException)
    {
      EndRequest("參數(shù)類型匹配發(fā)生錯誤!");
    }
    catch (InvalidCastException)
    {
      EndRequest("參數(shù)類型轉換發(fā)生錯誤!");
    }
    catch (ThreadAbortException)
    {
    }
    catch (Exception e)
    {
      EndRequest(e.Message);
    }
  }
  /// <summary>
  /// 獲取函數(shù)元數(shù)據(jù)并緩存
  /// </summary>
  /// <param name="methodName"></param>
  /// <returns></returns>
  private CacheMethodInfo TryGetMethodInfo(string methodName)
  {
    Type type = this.GetType().BaseType;
    string cacheKey = type.AssemblyQualifiedName;
    Dictionary<string, CacheMethodInfo> dic = _ajaxTable[cacheKey] as Dictionary<string, CacheMethodInfo>;
    if (dic == null)
    {
      dic = new Dictionary<string, CacheMethodInfo>();
      MethodInfo[] methodInfos = (from m in type.GetMethods(BindingFlags.Public | BindingFlags.Instance | BindingFlags.Static)
                    let ma = m.GetCustomAttributes(typeof(AjaxMethodAttribute), false)
                    where ma.Length > 0
                    select m).ToArray();
      foreach (var mi in methodInfos)
      {
        CacheMethodInfo cacheInfo = new CacheMethodInfo();
        cacheInfo.MethodName = mi.Name;
        cacheInfo.MethodInfo = mi;
        cacheInfo.Parameters = mi.GetParameters();
        dic.Add(mi.Name, cacheInfo);
      }
      _ajaxTable.Add(cacheKey, dic);
    }
    CacheMethodInfo targetInfo = null;
    dic.TryGetValue(methodName, out targetInfo);
    return targetInfo;
  }
  /// <summary>
  /// 獲取函數(shù)參數(shù)
  /// </summary>
  /// <param name="parameterInfos"></param>
  /// <returns></returns>
  private object[] GetParameters(ParameterInfo[] parameterInfos)
  {
    if (parameterInfos == null || parameterInfos.Length <= 0)
    {
      return null;
    }
    HttpRequest request = this.Context.Request;
    NameValueCollection nvc = null;
    string requestType = request.RequestType;
    if (string.Compare("GET", requestType, true) == 0)
    {
      nvc = request.QueryString;
    }
    else
    {
      nvc = request.Form;
    }
    int length = parameterInfos.Length;
    object[] parameters = new object[length];
    if (nvc == null || nvc.Count <= 0)
    {
      return parameters;
    }
    for (int i = 0; i < length; i++)
    {
      ParameterInfo pi = parameterInfos[i];
      string[] values = nvc.GetValues(pi.Name);
      object value = null;
      if (values != null)
      {
        if (values.Length > 1)
        {
          value = String.Join(",", values);
        }
        else
        {
          value = values[0];
        }
      }
      if (value == null)
      {
        continue;
      }
      parameters[i] = Convert.ChangeType(value, pi.ParameterType);
    }      
    return parameters;
  }
  private void EndRequest(string msg)
  {
    HttpResponse response = this.Context.Response;
    response.Write(msg);
    response.End();
  }
}

  頁面類:

public string GetData3(int i, double d, string str)
{
  string[] datas = new string[] { i.ToString(), d.ToString(), str };
  return "參數(shù)分別是:" + String.Join(",", datas);
}

  前臺代碼:

function getData3(){
  $.ajax({
    headers:{"AjaxFlag":"XHR","MethodName":"GetData3"},
    data:{"i":1,"d":"10.1a","str":"hehe"},
    success:function(data){
      $("#result").text(data);
    }
  });
}

五、總結

  上面的頁面基類已經(jīng)具備可以完成基本的功能,但它還不夠好。主要有:

  1. 依附在頁面基類。對于本來有頁面基類的,無疑會變得更加復雜。我們希望把它獨立開來,變成一個單獨的組件。

  2. 效率問題。反射的效率是很低的,尤其在web這類應用程序上,更應該慎用。以動態(tài)執(zhí)行函數(shù)為例,效率主要低在:a.根據(jù)字符串動態(tài)查找函數(shù)的過程。b.執(zhí)行函數(shù)時,反射內(nèi)部需要將參數(shù)打包成一個數(shù)組,再將參數(shù)解析到線程棧上;在調(diào)用前CLR還要檢測參數(shù)的正確性,再判斷有沒有權限執(zhí)行。上面的優(yōu)化其實只優(yōu)化了一半,也就是優(yōu)化了查找的過程,而Invoke同樣會有性能損失。當然,隨著.net版本越高,反射的效率也會有所提升,但這種動態(tài)的東西,始終是用效率換取靈活性的。

  3.不能支持復雜參數(shù)。有時候參數(shù)比較多,函數(shù)參數(shù)一般會封裝成一個對象類型。

  4. AjaxMethodAttribute只是一個空的標記屬性。我們可以為它加入一些功能,例如,標記函數(shù)的名稱、是否使用Session、緩存設置等都可以再這里完成。

  用過WebForm的朋友可能會提到AjaxPro組件,這是一個開源的組件,下一篇就通過源碼了解這個組件,借鑒它的處理過程,并且分析它的優(yōu)缺點。

關于編寫輕量ajax組件01-與webform平臺上的各種實現(xiàn)方式比較的示例分析問題的解答就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關注億速云行業(yè)資訊頻道了解更多相關知識。

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權內(nèi)容。

AI