模拟IL 指令输出等效的C#

public static void ParameterSetter_User(IDbCommand command, T obj)
    {
        var createParameterMethod = command.GetType().GetMethod("CreateParameter");
        var addMethod = command.Parameters.GetType().GetMethod("Add", new[] { typeof(SqlParameter) });

        foreach (var property in propertyList)
        {
            var parameter = (IDataParameter)createParameterMethod.Invoke(command, null);

            parameter.ParameterName = $"@{property.Name.ToLower()}";
            var value = property.GetValue(obj);
            parameter.Value = value ?? DBNull.Value;

            command.Parameters.Add(parameter);
        }
    }

ParameterSetter_User 方法的实现转换成使用 Emit 生成IL代码的方式如下所示:

public static void ParameterSetter_User(IDbCommand command, T obj)
{
    var dynamicMethod = new DynamicMethod($"ParameterSetter_{typeof(T).Name}", null, new[] { typeof(IDbCommand), typeof(T) }, typeof(T), true);
    var ilGenerator = dynamicMethod.GetILGenerator();

    var createParameterMethod = typeof(IDbCommand).GetMethod("CreateParameter");
    var addMethod = typeof(SqlParameterCollection).GetMethod("Add", new[] { typeof(SqlParameter) });

    // Declare local variables
    var parameterLocal = ilGenerator.DeclareLocal(typeof(IDataParameter));
    var valueLocal = ilGenerator.DeclareLocal(typeof(object));

    foreach (var property in propertyList)
    {
        // Create a new parameter
        ilGenerator.Emit(OpCodes.Ldarg_0); // Load command
        ilGenerator.Emit(OpCodes.Callvirt, createParameterMethod); // Call CreateParameter
        ilGenerator.Emit(OpCodes.Stloc, parameterLocal); // Store IDataParameter in local variable

        // Set ParameterName property
        ilGenerator.Emit(OpCodes.Ldloc, parameterLocal); // Load parameter
        ilGenerator.Emit(OpCodes.Ldstr, $"@{property.Name.ToLower()}"); // Load parameter name
        ilGenerator.Emit(OpCodes.Callvirt, typeof(IDataParameter).GetProperty("ParameterName").GetSetMethod()); // Set ParameterName

        // Get property value
        ilGenerator.Emit(OpCodes.Ldarg_1); // Load obj (instance of T)
        ilGenerator.Emit(OpCodes.Callvirt, property.GetGetMethod()); // Call property getter
        ilGenerator.Emit(OpCodes.Stloc, valueLocal); // Store property value in local variable

        // Set Value property
        ilGenerator.Emit(OpCodes.Ldloc, parameterLocal); // Load parameter
        ilGenerator.Emit(OpCodes.Ldloc, valueLocal); // Load property value
        ilGenerator.Emit(OpCodes.Ldnull); // Load DBNull.Value
        ilGenerator.Emit(OpCodes.Ceq); // Compare with null
        var valueNotNullLabel = ilGenerator.DefineLabel();
        ilGenerator.Emit(OpCodes.Brfalse_S, valueNotNullLabel); // Branch if not null
        ilGenerator.Emit(OpCodes.Pop); // Pop comparison result
        ilGenerator.Emit(OpCodes.Ldsfld, typeof(DBNull).GetField("Value")); // Load DBNull.Value
        var valueSetLabel = ilGenerator.DefineLabel();
        ilGenerator.Emit(OpCodes.Br_S, valueSetLabel); // Branch to set value
        ilGenerator.MarkLabel(valueNotNullLabel); // Mark label for value not null
        ilGenerator.Emit(OpCodes.Ldloc, valueLocal); // Load property value
        ilGenerator.MarkLabel(valueSetLabel); // Mark label for setting value
        ilGenerator.Emit(OpCodes.Callvirt, typeof(IDataParameter).GetProperty("Value").GetSetMethod()); // Set Value

        // Add parameter to command's Parameters collection
        ilGenerator.Emit(OpCodes.Ldarg_0); // Load command
        ilGenerator.Emit(OpCodes.Callvirt, addMethod); // Call Add method of Parameters collection
        ilGenerator.Emit(OpCodes.Pop); // Pop result of Add method
    }

    ilGenerator.Emit(OpCodes.Ret); // Return from method

    // Create delegate for parameterSetter
    var setterDelegate = (Action<IDbCommand, T>)dynamicMethod.CreateDelegate(typeof(Action<IDbCommand, T>));
    setterDelegate(command, obj); // Invoke the generated method
}

这段代码使用 DynamicMethodILGenerator 生成IL代码,实现了和之前的反射版本功能相同的动态方法 ParameterSetter_User。主要步骤包括:

  1. 声明动态方法和ILGenerator: 使用 DynamicMethod 创建动态方法,并获取其 ILGenerator

  2. 获取方法和添加方法准备: 使用 typeof(IDbCommand).GetMethod("CreateParameter")typeof(SqlParameterCollection).GetMethod("Add") 获取方法信息。

  3. 循环遍历属性列表: 对于每个属性,依次生成IL指令实现:

    • 调用 CreateParameter 方法创建新的参数对象。
    • 设置 ParameterName 属性为属性名称的小写形式。
    • 获取属性值,并根据是否为null设置参数的 Value 属性为属性值或者 DBNull.Value
    • 将参数添加到命令的参数集合中,使用 Add 方法。
  4. 生成完毕: 最后生成 Ret 指令表示方法结束,创建委托并调用生成的方法。

这种方式避免了反射的性能开销,并允许更精细地控制生成的IL代码逻辑。

  • 5
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值