属性值动态获取和赋值(反射、表达式、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
{
    ///


    /// 属性值动态获取和赋值(get、set)
    ///

    public class PropertyUtil
    {
        ///
        /// 反射获取对象的属性值
        ///

        ///
        ///
        ///
        public static object ReflectGetter(object obj, string propertyName)
        {
            var type = obj.GetType();
            var propertyInfo = type.GetProperty(propertyName);
            var propertyValue = propertyInfo.GetValue(obj);
            return propertyValue;
        }


        ///
        /// 反射设置对象的属性值
        ///

        ///
        ///
        ///
        public static void ReflectSetter(object obj, string propertyName, object propertyValue)
        {
            var type = obj.GetType();
            var propertyInfo = type.GetProperty(propertyName);
            propertyInfo.SetValue(obj, propertyValue);
        }


        ///
        /// 表达式获取对象的属性值
        ///

        ///
        ///
        ///
        public static Func ExpresionGetter(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>(callMethod, parameterExpression);


            return expression.Compile();
        }


        ///
        /// 表达式设置对象的属性值
        ///

        ///
        ///
        ///
        public static Action ExpresionSetter(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>(body, objectParameterExpression, valueParameterExpression);


            return expression.Compile();
        }


        ///
        /// Emit获取对象的属性值
        ///

        ///
        ///
        ///
        public static Func EmitGetter(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)) as Func;
        }


        ///
        /// Emit设置对象的属性值
        ///

        ///
        ///
        ///
        public static Action EmitSetter(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)) as Action;
        }
    }
}


单元测试

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
    {
        ///


        /// 
        ///

        public class Logic
        {
            ///
            /// 
            ///

            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("Name");
            setterMethod(model, "测试");


            // 获取 model 的 Name 字段值
            var getterMethod = PropertyUtil.ExpresionGetter("Name");
            var value = getterMethod(model);


            Assert.IsTrue(value.ToString() == "测试");
        }


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


            // 设置 model 的 Name 字段值 = 测试
            var setterMethod = PropertyUtil.EmitSetter("Name");
            setterMethod(model, "测试");


            // 获取 model 的 Name 字段值
            var getterMethod = PropertyUtil.EmitGetter("Name");
            var value = getterMethod(model);


            Assert.IsTrue(value.ToString() == "测试");
        }
    }
}


你可能感兴趣的:(.net)