91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

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

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

.NET MyMVC框架給方法賦值的案例

發布時間:2020-08-28 14:03:38 來源:億速云 閱讀:118 作者:小新 欄目:編程語言

這篇文章主要介紹.NET MyMVC框架給方法賦值的案例,文中介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!

GetActionCallParameters的實現過程:

private static object[] GetActionCallParameters(HttpContext context, ActionDescription action)
{    if( action.Parameters == null || action.Parameters.Length == 0 )        return null;    object[] parameters = new object[action.Parameters.Length];    for( int i = 0; i < action.Parameters.Length; i++ ) {        ParameterInfo p = action.Parameters[i];        if( p.IsOut )            continue;        if( p.ParameterType == typeof(NameValueCollection) ) {            if( string.Compare(p.Name, "Form", StringComparison.OrdinalIgnoreCase) == 0 )
                parameters[i] = context.Request.Form;            else if( string.Compare(p.Name, "QueryString", StringComparison.OrdinalIgnoreCase) == 0 )
                parameters[i] = context.Request.QueryString;            else if( string.Compare(p.Name, "Headers", StringComparison.OrdinalIgnoreCase) == 0 )
                parameters[i] = context.Request.Headers;            else if( string.Compare(p.Name, "ServerVariables", StringComparison.OrdinalIgnoreCase) == 0 )
                parameters[i] = context.Request.ServerVariables;
        }        else{            Type paramterType = p.ParameterType.GetRealType();            // 如果參數是簡單類型,則直接從HttpRequest中讀取并賦值            if( paramterType.IsSimpleType() ) {                object val = ModelHelper.GetValueByKeyAndTypeFrommRequest(
                                                context.Request, p.Name, paramterType, null);                if( val != null )
                    parameters[i] = val;
            }            else {                // 自定義的類型。首先創建實例,然后給所有成員賦值。
                // 注意:這里不支持嵌套類型的自定義類型。                object item = Activator.CreateInstance(paramterType);                ModelHelper.FillModel(context.Request, item, p.Name);
                parameters[i] = item;
            }
        }
    }    return parameters;
}

要理解這段代碼還要從前面的【查找Action的過程】說起,在那個階段,可以獲取一個Action的描述,具體在框架內部表示為ActionDescription類型:

internal sealed class ActionDescription : BaseDescription{    public ControllerDescription PageController; //為PageAction保留    public MethodInfo MethodInfo { get; private set; }    public ActionAttribute Attr { get; private set; }    public ParameterInfo[] Parameters { get; private set; }    public bool HasReturn { get; private set; }    public ActionDescription(MethodInfo m, ActionAttribute atrr) : base(m)
    {        this.MethodInfo = m;        this.Attr = atrr;        this.Parameters = m.GetParameters();        this.HasReturn = m.ReturnType != ReflectionHelper.VoidType;
    }
}

在構造函數的第三行代碼中,我就可以得到這個方法的所有參數情況。
然后,我在就可以在GetActionCallParameters方法中,循環每個參數的定義,為它們賦值。
這段代碼也解釋了前面所說的只支持4種NameValueCollection集合的原因。

注意了,我在獲取每個參數的類型時,是使用了下面的語句:

Type paramterType = p.ParameterType.GetRealType();

實際上,ParameterType就已經反映了參數的類型,為什么不直接使用它呢?
答:因為【可空泛型】的原因。這個類型我們需要特殊的處理。
例如:如果某個參數是這樣聲明的: int? id
那么,即使在QueryString中包含id這樣一個參數,我也不能直接轉成 int? 使用這種類型,必須得到它的【實際類型】。
GetRealType()是個擴展方法,它就專門完成這個功能:

/// <summary>
/// 得到一個實際的類型(排除Nullable類型的影響)。比如:int? 最后將得到int/// </summary>
/// <param name="type"></param>
/// <returns></returns>public static Type GetRealType(this Type type)
{    if( type.IsGenericType )        return Nullable.GetUnderlyingType(type) ?? type;    else
        return type;
}

如果某個參數的類型是一個自定義的類型,框架會先創建實例(調用無參的構造函數),然后給它的Property, Field賦值。

注意了:自定義的類型,一定要提供一個無參的構造函數。

為自定義類型的實例填充數據成員的代碼如下:

internal static class ModelHelper{    public static readonly bool IsDebugMode;    static ModelHelper()
    {        CompilationSection configSection = 
                    ConfigurationManager.GetSection("system.web/compilation") as CompilationSection;        if( configSection != null )
            IsDebugMode = configSection.Debug;
    }    /// <summary>
    /// 根據HttpRequest填充一個數據實體。    /// 這里不支持嵌套類型的數據實體,且要求各數據成員都是簡單的數據類型。    /// </summary>
    /// <param name="request"></param>
    /// <param name="model"></param>    public static void FillModel(HttpRequest request, object model, string paramName)
    {        ModelDescripton descripton = ReflectionHelper.GetModelDescripton(model.GetType());        object val = null;        foreach( DataMember field in descripton.Fields ) {            // 這里的實現方式不支持嵌套類型的數據實體。
            // 如果有這方面的需求,可以將這里改成遞歸的嵌套調用。            val = GetValueByKeyAndTypeFrommRequest(
                                request, field.Name, field.Type.GetRealType(), paramName);            if( val != null )
                field.SetValue(model, val);
        }
    }    /// <summary>
    /// 讀取一個HTTP參數值。這里只讀取QueryString以及Form    /// </summary>
    /// <param name="request"></param>
    /// <param name="key"></param>
    /// <returns></returns>    public static string GetHttpValue(HttpRequest request, string key)
    {        string val = request.QueryString[key];        if( val == null )
            val = request.Form[key];        return val;
    }    
    public static object GetValueByKeyAndTypeFrommRequest(                        HttpRequest request, string key, Type type, string paramName)
    {        // 不支持復雜類型        if( type.IsSimpleType() == false )            return null;        string val = GetHttpValue(request, key);        if( val == null ) {            // 再試一次。有可能是多個自定義類型,Form表單元素采用變量名做為前綴。            if( string.IsNullOrEmpty(paramName) == false ) {
                val = GetHttpValue(request, paramName + "." + key);
            }            if( val == null )                return null;
        }        return SafeChangeType(val.Trim(), type);
    }    public static object SafeChangeType(string value, Type conversionType)
    {        if( conversionType == typeof(string) )            return value;        if( value == null || value.Length == 0 )            // 空字符串根本不能做任何轉換,所以直接返回null            return null;        try {            // 為了簡單,直接調用 .net framework中的方法。
            // 如果轉換失敗,則會拋出異常。            return Convert.ChangeType(value, conversionType);
        }        catch {            if( IsDebugMode )                throw;            // Debug 模式下拋異常            else
                return null;    // Release模式下忽略異常(防止惡意用戶錯誤輸入)        }
    }
}

在給自定義的數據類型實例加載數據前,需要先知道這個實例對象有哪些屬性以及字段,這個過程的代碼如下:

/// <summary>
/// 返回一個實體類型的描述信息(全部屬性及字段)。/// </summary>
/// <param name="type"></param>
/// <returns></returns>public static ModelDescripton GetModelDescripton(Type type)
{    if( type == null )        throw new ArgumentNullException("type");    
    string key = type.FullName;    ModelDescripton mm = (ModelDescripton)s_modelTable[key];    if( mm == null ) {        List<DataMember> list = new List<DataMember>();
        (from p in type.GetProperties(BindingFlags.Instance | BindingFlags.Public)         select new PropertyMember(p)).ToList().ForEach(x=>list.Add(x));
        (from f in type.GetFields(BindingFlags.Instance | BindingFlags.Public)         select new FieldMember(f)).ToList().ForEach(x => list.Add(x));
        mm = new ModelDescripton { Fields = list.ToArray() };
        s_modelTable[key] = mm;
    }    return mm;
}

在拿到一個類型的所有屬性以及字段的描述信息后,就可以通過循環的方式,根據這些數據成員的名字去QueryString,Form讀取所需的數據了。

以上是.NET MyMVC框架給方法賦值的案例的所有內容,感謝各位的閱讀!希望分享的內容對大家有幫助,更多相關知識,歡迎關注億速云行業資訊頻道!

向AI問一下細節

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

ne
AI

玉树县| 疏附县| 新密市| 蒙城县| 西畴县| 襄城县| 北宁市| 旬邑县| 太和县| 叶城县| 信阳市| 定日县| 盐城市| 岳阳市| 阿图什市| 凌云县| 泽普县| 临猗县| 兴化市| 上思县| 雷山县| 巩留县| 琼中| 清涧县| 柯坪县| 广元市| 浠水县| 连州市| 梅河口市| 肇东市| 昂仁县| 玉门市| 柳州市| 冀州市| 宁南县| 巴林左旗| 伊宁市| 文成县| 屏东市| 勃利县| 蒙阴县|