1. 程式人生 > >屬性值動態獲取和賦值(反射、表示式、Emit)

屬性值動態獲取和賦值(反射、表示式、Emit)

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Reflection;
using System.Reflection.Emit;

using System.Linq.Expressions;

namespace Util
{
    /// <summary>
    /// 屬性值動態獲取和賦值(get、set)
    /// </summary>
    public class PropertyUtil
    {
        /// <summary>
        /// 反射獲取物件的屬性值
        /// </summary>
        /// <param name="obj"></param>
        /// <param name="propertyName"></param>
        /// <returns></returns>
        public static object ReflectGetter(object obj, string propertyName)
        {
            var type = obj.GetType();
            var propertyInfo = type.GetProperty(propertyName);
            var propertyValue = propertyInfo.GetValue(obj);
            return propertyValue;
        }


        /// <summary>
        /// 反射設定物件的屬性值
        /// </summary>
        /// <param name="obj"></param>
        /// <param name="propertyName"></param>
        /// <param name="propertyValue"></param>
        public static void ReflectSetter(object obj, string propertyName, object propertyValue)
        {
            var type = obj.GetType();
            var propertyInfo = type.GetProperty(propertyName);
            propertyInfo.SetValue(obj, propertyValue);
        }


        /// <summary>
        /// 表示式獲取物件的屬性值
        /// </summary>
        /// <typeparam name="T"></typeparam>
        /// <param name="propertyName"></param>
        /// <returns></returns>
        public static Func<T, object> ExpresionGetter<T>(string propertyName)
        {
            var type = typeof(T);
            var property = type.GetProperty(propertyName);


            //// 物件例項
            var parameterExpression = Expression.Parameter(typeof(object), "obj");


            //// 轉換引數為真實型別
            var unaryExpression = Expression.Convert(parameterExpression, type);


            //// 呼叫獲取屬性的方法
            var callMethod = Expression.Call(unaryExpression, property.GetGetMethod());
            var expression = Expression.Lambda<Func<T, object>>(callMethod, parameterExpression);


            return expression.Compile();
        }


        /// <summary>
        /// 表示式設定物件的屬性值
        /// </summary>
        /// <typeparam name="T"></typeparam>
        /// <param name="propertyName"></param>
        /// <returns></returns>
        public static Action<T, object> ExpresionSetter<T>(string propertyName)
        {
            var type = typeof(T);
            var property = type.GetProperty(propertyName);


            var objectParameterExpression = Expression.Parameter(typeof(object), "obj");
            var objectUnaryExpression = Expression.Convert(objectParameterExpression, type);


            var valueParameterExpression = Expression.Parameter(typeof(object), "val");
            var valueUnaryExpression = Expression.Convert(valueParameterExpression, property.PropertyType);


            //// 呼叫給屬性賦值的方法
            var body = Expression.Call(objectUnaryExpression, property.GetSetMethod(), valueUnaryExpression);
            var expression = Expression.Lambda<Action<T, object>>(body, objectParameterExpression, valueParameterExpression);


            return expression.Compile();
        }


        /// <summary>
        /// Emit獲取物件的屬性值
        /// </summary>
        /// <typeparam name="T"></typeparam>
        /// <param name="propertyName"></param>
        /// <returns></returns>
        public static Func<T, object> EmitGetter<T>(string propertyName)
        {
            var type = typeof(T);


            var dynamicMethod = new DynamicMethod("get_" + propertyName, typeof(object), new[] { type }, type);
            var iLGenerator = dynamicMethod.GetILGenerator();
            iLGenerator.Emit(OpCodes.Ldarg_0);


            var property = type.GetProperty(propertyName);
            iLGenerator.Emit(OpCodes.Callvirt, property.GetMethod);


            if (property.PropertyType.IsValueType)
            {
                // 如果是值型別,裝箱
                iLGenerator.Emit(OpCodes.Box, property.PropertyType);
            }
            else
            {
                // 如果是引用型別,轉換
                iLGenerator.Emit(OpCodes.Castclass, property.PropertyType);
            }


            iLGenerator.Emit(OpCodes.Ret);


            return dynamicMethod.CreateDelegate(typeof(Func<T, object>)) as Func<T, object>;
        }


        /// <summary>
        /// Emit設定物件的屬性值
        /// </summary>
        /// <typeparam name="T"></typeparam>
        /// <param name="propertyName"></param>
        /// <returns></returns>
        public static Action<T, object> EmitSetter<T>(string propertyName)
        {
            var type = typeof(T);


            var dynamicMethod = new DynamicMethod("EmitCallable", null, new[] { type, typeof(object) }, type.Module);
            var iLGenerator = dynamicMethod.GetILGenerator();


            var callMethod = type.GetMethod("set_" + propertyName, BindingFlags.Instance | BindingFlags.IgnoreCase | BindingFlags.Public);
            var parameterInfo = callMethod.GetParameters()[0];
            var local = iLGenerator.DeclareLocal(parameterInfo.ParameterType, true);


            iLGenerator.Emit(OpCodes.Ldarg_1);
            if (parameterInfo.ParameterType.IsValueType)
            {
                // 如果是值型別,拆箱
                iLGenerator.Emit(OpCodes.Unbox_Any, parameterInfo.ParameterType);
            }
            else
            {
                // 如果是引用型別,轉換
                iLGenerator.Emit(OpCodes.Castclass, parameterInfo.ParameterType);
            }


            iLGenerator.Emit(OpCodes.Stloc, local);
            iLGenerator.Emit(OpCodes.Ldarg_0);
            iLGenerator.Emit(OpCodes.Ldloc, local);


            iLGenerator.EmitCall(OpCodes.Callvirt, callMethod, null);
            iLGenerator.Emit(OpCodes.Ret);


            return dynamicMethod.CreateDelegate(typeof(Action<T, object>)) as Action<T, object>;
        }
    }
}

單元測試

using Util;
using Microsoft.VisualStudio.TestTools.UnitTesting;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;


namespace Tests.Util
{
    [TestClass]
    public class PropertyUtil_Test
    {
        /// <summary>
        /// 
        /// </summary>
        public class Logic
        {
            /// <summary>
            /// 
            /// </summary>
            public string Name { get; set; }


        }


        [TestMethod]
        public void Reflect_Test()
        {
            var model = new Logic();


            // 設定 model 的 Name 欄位值 = 測試
            PropertyUtil.ReflectSetter(model, "Name", "測試");


            // 獲取 model 的 Name 欄位值
            var value = PropertyUtil.ReflectGetter(model, "Name");


            Assert.IsTrue(value.ToString() == "測試");
        }


        [TestMethod]
        public void Expresion_Test()
        {
            var model = new Logic();


            // 設定 model 的 Name 欄位值 = 測試
            var setterMethod = PropertyUtil.ExpresionSetter<Logic>("Name");
            setterMethod(model, "測試");


            // 獲取 model 的 Name 欄位值
            var getterMethod = PropertyUtil.ExpresionGetter<Logic>("Name");
            var value = getterMethod(model);


            Assert.IsTrue(value.ToString() == "測試");
        }


        [TestMethod]
        public void Emit_Test()
        {
            var model = new Logic();


            // 設定 model 的 Name 欄位值 = 測試
            var setterMethod = PropertyUtil.EmitSetter<Logic>("Name");
            setterMethod(model, "測試");


            // 獲取 model 的 Name 欄位值
            var getterMethod = PropertyUtil.EmitGetter<Logic>("Name");
            var value = getterMethod(model);


            Assert.IsTrue(value.ToString() == "測試");
        }
    }
}