MVC源碼分析 - ModelBinder綁定 / 自定義數據綁定

来源:http://www.cnblogs.com/elvinle/archive/2017/01/20/6307136.html
-Advertisement-
Play Games

這幾天老感覺不對, 總覺得少點什麼, 今天才發現, 前面 3 裡面, 在獲取Action參數信息的時候, 少解析了. 裡面還有一個比較重要的東西. 今天看也是一樣的. 在 InvokeAction() 方法裡面, 有一句代碼: 這個是用來獲取參數的. 那麼參數是不是隨便獲取呢? 在Mvc 裡面, 頁 ...


這幾天老感覺不對, 總覺得少點什麼, 今天才發現, 前面 3 裡面, 在獲取Action參數信息的時候,  少解析了. 裡面還有一個比較重要的東西. 今天看也是一樣的.

在 InvokeAction() 方法裡面, 有一句代碼:

IDictionary<string, object> parameterValues = this.GetParameterValues(controllerContext, actionDescriptor);

這個是用來獲取參數的. 那麼參數是不是隨便獲取呢? 在Mvc 裡面, 頁面向Action 傳參的時候, 有沒有嘗試過傳一個數組, 然後接收的時候, 也直接解析成數組呢? 或者接收更複雜的類型呢?

答案都在這一篇裡面了. 先來看源碼.

protected virtual IDictionary<string, object> GetParameterValues(ControllerContext controllerContext,
   ActionDescriptor actionDescriptor) { Dictionary
<string, object> dictionary = new Dictionary<string, object>(StringComparer.OrdinalIgnoreCase); foreach (ParameterDescriptor descriptor in actionDescriptor.GetParameters()) { dictionary[descriptor.ParameterName] = this.GetParameterValue(controllerContext, descriptor); } return dictionary; }

一、源碼解析

1. actionDescriptor.GetParameters()

//System.Web.Mvc.ReflectedActionDescriptor
public override ParameterDescriptor[] GetParameters()
{
    return ActionDescriptorHelper.GetParameters(this, this.MethodInfo, ref this._parametersCache);
}

這裡應該是獲取所有的參數和其描述信息.

 

2. this.GetParameterValue() -- 主要方法

protected virtual object GetParameterValue(ControllerContext controllerContext,
   ParameterDescriptor parameterDescriptor) { Type parameterType
= parameterDescriptor.ParameterType;
   //根據參數描述來獲取參數的處理介面 IModelBinder modelBinder
= this.GetModelBinder(parameterDescriptor); IValueProvider valueProvider = controllerContext.Controller.ValueProvider; string str = parameterDescriptor.BindingInfo.Prefix ?? parameterDescriptor.ParameterName;
   //獲取參數上面的過濾器, 併在下麵放入到參數解析上下文中(ModelBindingContext) Predicate
<string> propertyFilter = GetPropertyFilter(parameterDescriptor); ModelBindingContext bindingContext = new ModelBindingContext { FallbackToEmptyPrefix = parameterDescriptor.BindingInfo.Prefix == null, ModelMetadata = ModelMetadataProviders.Current.GetMetadataForType(null, parameterType), ModelName = str, ModelState = controllerContext.Controller.ViewData.ModelState, PropertyFilter = propertyFilter, ValueProvider = valueProvider };
   //執行參數的處理程式
return (modelBinder.BindModel(controllerContext, bindingContext) ?? parameterDescriptor.DefaultValue); }

 

2.1 GetModelBinder()方法

private IModelBinder GetModelBinder(ParameterDescriptor parameterDescriptor)
{
    return (parameterDescriptor.BindingInfo.Binder ?? 
      this.Binders.GetBinder(parameterDescriptor.ParameterType)); }

這裡是根據參數描述來獲取參數的處理介面

public interface IModelBinder
{
    // Methods
    object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext);
}

 

2.2 BindModel()方法 - 這裡看的是 DefaultModelBinder, 後面會自定義一個

public virtual object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
{
    EnsureStackHelper.EnsureStack();
    if (bindingContext == null)
    {
        throw new ArgumentNullException("bindingContext");
    }
    bool flag = false;
    if (!string.IsNullOrEmpty(bindingContext.ModelName) 
    && !bindingContext.ValueProvider.ContainsPrefix(bindingContext.ModelName)) { if (!bindingContext.FallbackToEmptyPrefix) { return null; } ModelBindingContext context = new ModelBindingContext { ModelMetadata = bindingContext.ModelMetadata, ModelState = bindingContext.ModelState, PropertyFilter = bindingContext.PropertyFilter, ValueProvider = bindingContext.ValueProvider }; bindingContext = context; flag = true; } if (!flag) { bool flag2 = ShouldPerformRequestValidation(controllerContext, bindingContext); bool skipValidation = !flag2; ValueProviderResult valueProviderResult = bindingContext.UnvalidatedValueProvider
        .GetValue(bindingContext.ModelName, skipValidation);
if (valueProviderResult != null) {
       //為簡單對象返回參數值
return this.BindSimpleModel(controllerContext, bindingContext, valueProviderResult); } } if (!bindingContext.ModelMetadata.IsComplexType) { return null; } return this.BindComplexModel(controllerContext, bindingContext); }

這裡面的內容有點多, 也有點複雜, 其實解析到這裡, 差不多已經得到我想要的東西了.

 

二、自定義ModelBinder

1. IModelBinder -- 對於相對比較簡單的類型, 可以使用這種方式

首先新建一個稍微複雜一點的類.

public enum GenderEnum
{ 
    Female,
    Male,
    Unknow
}

public class ExtInfo
{
    public string QQ { get; set; }
    public string Phone { get; set; }
}

public class User
{
   //這裡的構造函數, 我創建了一個ExtInfo實例, 否則, 一會這個ExtInfo對象就是空的, 影響我的演示
public User() { Extension = new ExtInfo(); }
public int Id { get; set; } public string Name { get; set; } public GenderEnum Gender { get; set; } public ExtInfo Extension { get; set; } }

接下來, 看一下控制器代碼

public class ModelController : Controller
{
    public ActionResult Get(User user)
    {
        return View(user);
    }

    public ActionResult Index([ModelBinder(typeof(ModelIndexBinder))]User user)
    {
        return View(user);
    }
}

控制器中需要註意的部分, 我已經標紅了. 接下來看我自定義的ModelBinder

public class ModelIndexBinder : IModelBinder
{
    public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
    {
        var request = controllerContext.HttpContext.Request;
        User user = new User
        {
            Id = Convert.ToInt32(request.Params["Id"]),
            Name = request.Params["Name"].ToString(),
            Gender = (GenderEnum)Convert.ToInt32(request.Params["Gender"]),
            Extension = new ExtInfo
            {
                QQ = request.Params["QQ"].ToString(),
                Phone = request.Params["Phone"].ToString()
            }
        };
        return user;
    }
}

先看兩個方法執行的效果吧.

Index Get

 

 

 

註 : 這裡是可以有別的方式來達到目的的, 只需要修改一下url即可:

http://localhost:11620/model/Get?id=1&name=haha&gender=0&Extension.qq=123123123&Extension.phone=12312341234

此處是為了舉一個例子, 才這麼弄的.

對於比較簡單的, 用IModelBinder挺方便的, 但是對於稍微複雜一點的, 可以實現 DefaultModelBinder 來實現.

不過這部分實現就不解析了, 我暫時沒用過. 

一般對於Url Get請求, 不會很複雜, 只有在使用Post請求的時候, 會遇到比較複雜的處理方式. 但是對於這種複雜的處理, 還有一種比較常用的方式, 就是採用反序列化的方式, 我常用的是 Newtonsoft.

 

參考:

  MVC擴展

  深入Asp.net Mvc

 

 目錄已同步


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前言 使用普通用戶sudo echo 執行重定向命令的時候提示許可權不夠,已經在/etc/sudoers下做了配置。 www ALL=(ALL) NOPASSWD: /usr/bin/echo # 解決辦法 在使用sudo echo ‘hehe’>/usr/local/index.html的時候,其實 ...
  • Shell模板 ...
  • [root@localhost ~]# root——用戶名 localhost——本地 ~——家目錄 不同用戶不同 #——當前用戶是管理員 $——普通用戶 命令格式: 命令 [選項] [參數] pwd——當前目錄 cd [參數] 進入當前用戶的家 cd ~/cd [Enter] 進入上次目錄 cd ...
  • 大家都知道,現在和以前比起來,互聯網行業、軟體行業已經天差地別了。現在處處都在搞信息化建設,人人都知道互聯網思維。這樣的信息化時代,對於軟體開發者、對於軟體開發公司來說,是一個巨大的機遇。 在門外漢看來,軟體開發是機遇大、成本低,只要叫幾個程式員,就能搞出個軟體公司來。但是,事實情況是這個樣子嗎?本 ...
  • 在webuploader上傳大文件時必須配置一下,不然請求後臺處理程式時,會請求超時。出現404! <system.web> <httpRuntime maxRequestLength="2040000" useFullyQualifiedRedirectUrl="true" executionTi ...
  • 在 Windows Forms 和 WPF 應用中使用 FontAwesome 圖標 ...
  • 在C#中 “\”是特殊字元,要表示它的話需要使用“\\”。由於這種寫法不方便,C#語言提供了@對其簡化。只要在字元串前加上@即可直接使用“\”。所以上面的路徑在C#中應該表示為“Book”,@“\Tmp\Book”,@“C:\Tmp\Book”。 相對路徑使用“/”字元作為目錄的分隔字元,而絕對路徑 ...
  • 前言 上一篇講述了執行sql和配置的一些功能,這篇說明IQueryable(linq)或執行sql的查詢緩存與清理,包括擴展到將緩存存儲到Redis中。 擴展類庫源碼: github:https://github.com/skigs/EFCoreExtend 引用類庫: nuget:https:// ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...