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
}
这段代码使用 DynamicMethod
和 ILGenerator
生成IL代码,实现了和之前的反射版本功能相同的动态方法 ParameterSetter_User
。主要步骤包括:
-
声明动态方法和ILGenerator: 使用
DynamicMethod
创建动态方法,并获取其ILGenerator
。 -
获取方法和添加方法准备: 使用
typeof(IDbCommand).GetMethod("CreateParameter")
和typeof(SqlParameterCollection).GetMethod("Add")
获取方法信息。 -
循环遍历属性列表: 对于每个属性,依次生成IL指令实现:
- 调用
CreateParameter
方法创建新的参数对象。 - 设置
ParameterName
属性为属性名称的小写形式。 - 获取属性值,并根据是否为null设置参数的
Value
属性为属性值或者DBNull.Value
。 - 将参数添加到命令的参数集合中,使用
Add
方法。
- 调用
-
生成完毕: 最后生成
Ret
指令表示方法结束,创建委托并调用生成的方法。
这种方式避免了反射的性能开销,并允许更精细地控制生成的IL代码逻辑。