前不久在做脚本引擎的时候遇到这样一个要求:
脚本调用脚本引擎中的一个方法,调用的方法名称在脚本中指定,出于可扩展性考虑,第3方可以通过加入新的插件来增加脚本能够调用的方法。
1. 脚本只会给出要调用的方法名称,而这个不是在编译期确定的,所以需要根据名称使用反射查找方法,这很简单。
2. 脚本向方法传递参数都是JSON数据类型,也就是说,需要根据调用方法的参数类型将脚本传递过来的JSON数据类型转换为C#数据类型,可以是基本类型,也可以是结构体或类等复合类型。
难题就在第2点,JavaScriptSerializer类中有如下方法反序列化一个JSON字符串。
public T Deserialize<T>( string input )
但它的泛形参数不能在编译期确定,因为参数类型可以是无限扩充的。所以必须通过反射查找到要调用的方法后,再根据要调用方法的参数类型来反序列化。
实现原理如下:
1. 脚本传递要调用的函数名称以及参数(JSON字符串格式)到脚本引擎
2. 脚本引擎根据函数名称查找到要调用的函数信息
3. 脚本引擎使用反射动态生成对JavaScriptSerializer. Deserialize<T>的调用,而其中的T由第2步的反射给出
4. 调用。
示例代码如下:
//首先需要一个动态方法的委托,这里以一个参数的函数示例。 // json就是参数JSON字符串 // RemoteSink是插件从统一接口继承并实现的类 private delegate object DynamicMethodDelegate(string json, RemoteSink remoteSink); // //----------------------完美的分割线------------------------------- // // 下面这些参数的传递过程隐去 string methodName = XXX; // 脚本传递过来的方法名 string paramter = YYY; // 脚本传递过来的JSON字符串 RemoteSink sink = ZZZ; // 插件派生出来的处理类 // //----------------------完美的分割线------------------------------- // Type type = sink.GetType(); // 根据函数名称查找方法 MethodInfo methodInfo = type.GetMethod(methodName, BindingFlags.DeclaredOnly | BindingFlags.Public | BindingFlags.Instance); if (methodInfo == null) throw new ArgumentException("Error, cannot find the specific method by name."); // 方法的参数信息 ParameterInfo[] paramInfos = methodInfo.GetParameters(); if (paramInfos == null || paramInfos.Length != 1) { throw new ArgumentException("Error, method requires more than one paramter."); } // 创建一个动态方法 Type[] newMethodArgs = { typeof(string), type }; DynamicMethod newMethod = new DynamicMethod("dynamicMethod" , typeof(object) , newMethodArgs , type.Module ); // 生成此动态方法的方法体 ILGenerator generator = newMethod.GetILGenerator(); generator.Emit(OpCodes.Ldarg_1); // 首先实例化一个JavaScriptSerializer对象 generator.Emit(OpCodes.Newobj, typeof(JavaScriptSerializer).GetConstructor(Type.EmptyTypes)); // 根据反射得来的参数类型进行反序列化 generator.Emit(OpCodes.Ldarg_0); generator.Emit(OpCodes.Call, typeof(JavaScriptSerializer).GetMethod("Deserialize", BindingFlags.Public | BindingFlags.Instance).MakeGenericMethod(paramInfos[0].ParameterType)); // 调用插件中找到的方法 generator.Emit(OpCodes.Callvirt, methodInfo); generator.Emit(OpCodes.Ret); // 创建此动态方法的一个委托 DynamicMethodDelegate dynamicMethodDel = (DynamicMethodDelegate)newMethod.CreateDelegate(typeof(DynamicMethodDelegate)); // 终于可以调用了。 dynamicMethodDel(paramter, sink);