msdn 中MethodBase.Invoke 方法 介绍中的坑
模块开发总结:
c#动态调用webservices
来自网络及使用心得。
msdn:
MethodBase.Invoke 方法 (Object, Object[])
使用指定的参数调用当前实例所表示的方法或构造函数。
public Object Invoke (
Object obj,
Object[] parameters
)
参数
- obj
-
对其调用方法或构造函数的对象。如果方法是静态的,则忽略此参数。如果构造函数是静态的,则此参数必须为 空引用(在 Visual Basic 中为 Nothing) 或定义该构造函数的类的实例。
- parameters
-
调用的方法或构造函数的参数列表。这是一个对象数组,这些对象与要调用的方法或构造函数的参数具有相同的数量、顺序和类型。如果没有任何参数,则parameters 应为 空引用(在 Visual Basic 中为 Nothing)。
如果此实例所表示的方法或构造函数采用 ref 参数(在 Visual Basic 中为 ByRef),使用此函数调用该方法或构造函数时,该参数不需要任何特殊属性。如果数组中的对象未用值来显式初始化,则该对象将包含该对象类型的默认值。对于引用类型的元素,该值为 空引用(在 Visual Basic 中为 Nothing)。对于值类型的元素,该值为 0、0.0 或 false,具体取决于特定的元素类型。
返回值
一个对象,包含被调用方法的返回值,如果调用的是构造函数,则为 空引用(在 Visual Basic 中为 Nothing)。
xfire中提供的webservices(创建过程见baidu,很多很好)
public String invokedUseArray(String[] msgArray){ for (int i = 0; i < msgArray.length; i++) { System.out.println(msgArray[i]); } return "1"; } public String invokedUseString(String title,String desc){ System.out.println(title+"\t"+desc); return "1"; }
c#调用
string webserviceUrl==http://localhost:8080/bomcbpinterface/services/CreateGevtIncidentService; string title =”title”; string desc =”description”; string methodNameArray=”invokedUseArray”; string[] argsArray={title,desc} object result1 = WebServiceHelper.InvokeWebService(webserviceUrl,null,methodNameArray,new object[]{argsArray});//容易出错,报参数个数不对的异常 string methodNameString=”invokedUseString” object result2 = WebServiceHelper.InvokeWebService(webserviceUrl, null, methodNameString,argsArray);//容易出错,报参数个数不对的异常 或 object result2 = WebServiceHelper.InvokeWebService(webserviceUrl, null, methodNameString,new object[]{title,desc});//容易出错,报参数个数不对的异常
Utility:
需要添加引用:System.Web.Service
需要将.net 4.0 framework client profile,我们应该将它改成 .net 4.0 framework(VS2010)
具体步骤:
1. 从目标 URL 下载 WSDL 数据。
2. 使用 ServiceDescription 创建和格式化 WSDL 文档文件。
3. 使用 ServiceDescriptionImporter 创建客户端代理类。
4. 使用 CodeDom 动态创建客户端代理类程序集。
5. 利用反射调用相关 WebService 方法。
具体代码:
class WebServiceHelper { #region InvokeWebService /// <summary> /// 动态调用web服务 /// </summary> /// <param name="url">webservice url:http://xxx/zzz </param> /// <param name="className">className</param> /// <param name="methodName">methodName</param> /// <param name="args">the parameters used by methodName</param> /// <returns>type:object.Genanally,the result of the invoke operation</returns> public static object InvokeWebService(string url, string className, string methodName, object[] args) { //string @namespace = "alarm.mobile.work.inter.bomc.boco.com"; string @namespace = GetNamespace(url); logger.Info(@namespace); if ((className == null) || (className == "")) { className = WebServiceHelper.GetWsClassName(url); logger.Info(className); } try { logger.Info(className); // 1. 使用 WebClient 下载 WSDL 信息。 WebClient wc = new WebClient(); //ServiceDescription类提供一种方法,以创建和格式化用于描述 XML Web services 的有效的 Web 服务描述语言 (WSDL) 文档文件,该文件是完整的,具有适当的命名空间、元素和特性。 无法继承此类。 //ServiceDescription.Read 方法 (Stream) 通过直接从 Stream实例加载 XML 来初始化ServiceDescription类的实例。 Stream stream = wc.OpenRead(url + "?WSDL"); // 2. 创建和格式化 WSDL 文档。 ServiceDescription sd = ServiceDescription.Read(stream); // 3. 创建客户端代理代理类。 //ServiceDescriptionImporter 类 公开一种为 XML Web services 生成客户端代理类的方法。 //ServiceDescriptionImporter.AddServiceDescription 方法 将指定的ServiceDescription添加到要导入的ServiceDescriptions值的集合中。 ServiceDescriptionImporter sdi = new ServiceDescriptionImporter(); sdi.ProtocolName = "soap"; sdi.AddServiceDescription(sd, "", ""); // 4. 使用 CodeDom 编译客户端代理类。 //CodeNamespace表示命名空间声明 CodeNamespace cn = new CodeNamespace(@namespace);// 为代理类添加命名空间,缺省为全局空间。 logger.Info(cn.Name); //生成客户端代理类代码 // CodeCompileUnit会提供一个CodeDOM程式圆形的容器,CodeCompileUnit含有一个集合, //可以储存含有CodeDOM原始程式码原形,专案参考的组件集合以及专案组件属性集合的CodeNamespace物件。 CodeCompileUnit ccu = new CodeCompileUnit(); ccu.Namespaces.Add(cn); sdi.Import(cn, ccu); // CSharpCodeProvider类提供存取C#程式码产生器和程式码编译器的执行个体。 CodeDomProvider provider = new CSharpCodeProvider(); //设定编译参数 //创建编译器的参数实例 CompilerParameters cplist = new CompilerParameters(); cplist.GenerateExecutable = false; cplist.GenerateInMemory = true; cplist.ReferencedAssemblies.Add("System.dll"); cplist.ReferencedAssemblies.Add("System.XML.dll"); cplist.ReferencedAssemblies.Add("System.Web.Services.dll"); cplist.ReferencedAssemblies.Add("System.Data.dll"); //编译代理类 //CompilerResults表示从编译器返回的编译结果。使用指定的编译器设定, //根据CodeCompileUnit物件之指定阵列所包含的System.CodeDom树状结构,编译一个组件。 CompilerResults cr = provider.CompileAssemblyFromDom(cplist, ccu); //5. 使用 Reflection 调用 WebService 。 if (true == cr.Errors.HasErrors) { System.Text.StringBuilder sb = new System.Text.StringBuilder(); foreach (System.CodeDom.Compiler.CompilerError ce in cr.Errors) { sb.Append(ce.ToString()); sb.Append(System.Environment.NewLine); } logger.Error(sb.ToString()); //throw new Exception(sb.ToString()); } //生成代理实例,并调用方法 System.Reflection.Assembly assembly = cr.CompiledAssembly; // 如果在前面为代理类添加了命名空间,此处需要将命名空间添加到类型前面。 Type t = assembly.GetType(@namespace + "." + className, true, true); //Type t = assembly.GetType(className, true, true); // Activator类包含特定的方法,用以在本地或从远程创建对象类型,或获取对现有远程对象的引用。 //无法继承此类Activator.CreateInstance 方法 使用与指定参数匹配程度最高的构造函数创建指定类型的实例。 object obj = Activator.CreateInstance(t); //MethodInfo 的实例可以通过调用GetMethods或者Type对象或派生自Type的对象的GetMethod方法来获取, //还可以通过调用表示泛型方法定义的 MethodInfo 的MakeGenericMethod方法来获取。 System.Reflection.MethodInfo mi = t.GetMethod(methodName); logger.Info(mi.DeclaringType); return mi.Invoke(obj, args); } catch (Exception ex) { logger.Error(ex); //throw new Exception(ex.InnerException.Message, new Exception(ex.InnerException.StackTrace)); return null; } } /// <summary> /// get the name of class name /// </summary> /// <param name="wsUrl">webservice url:http://xxx/zzz </param> /// <returns></returns> private static string GetWsClassName(string wsUrl) { string[] parts = wsUrl.Split('/'); string[] pps = parts[parts.Length - 1].Split('.'); return pps[0]; } /// <summary> /// get the name of namespace /// </summary> /// <param name="URL">webservice url:http://xxx/zzz </param> /// <returns></returns> private static string GetNamespace(String URL) { HttpWebRequest request = (HttpWebRequest)WebRequest.Create(URL + "?WSDL"); SetWebRequest(request); WebResponse response = request.GetResponse(); StreamReader sr = new StreamReader(response.GetResponseStream(), Encoding.UTF8); XmlDocument doc = new XmlDocument(); doc.LoadXml(sr.ReadToEnd()); sr.Close(); string ns= doc.SelectSingleNode("//@targetNamespace").Value; return ns.Split('/')[ns.Split('/').Length - 1]; } private static void SetWebRequest(HttpWebRequest request) { request.Credentials = CredentialCache.DefaultCredentials; request.Timeout = 10000; } #endregion }
#region InvokeWebService //动态调用web服务 public static object InvokeWebService(string url, string className, string methodName, object[] args) { string @namespace = GetNamespace(url); logger.Info(@namespace); if ((className == null) || (className == "")) { className = WebServiceHelper.GetWsClassName(url); logger.Info(className); } try { logger.Info(className); // 1. 使用 WebClient 下载 WSDL 信息。 WebClient wc = new WebClient(); //ServiceDescription类提供一种方法,以创建和格式化用于描述 XML Web services 的有效的 Web 服务描述语言 (WSDL) 文档文件,该文件是完整的,具有适当的命名空间、元素和特性。 无法继承此类。 //ServiceDescription.Read 方法 (Stream) 通过直接从 Stream实例加载 XML 来初始化ServiceDescription类的实例。 Stream stream = wc.OpenRead(url + "?WSDL"); // 2. 创建和格式化 WSDL 文档。 ServiceDescription sd = ServiceDescription.Read(stream); // 3. 创建客户端代理代理类。 //ServiceDescriptionImporter 类 公开一种为 XML Web services 生成客户端代理类的方法。 //ServiceDescriptionImporter.AddServiceDescription 方法 将指定的ServiceDescription添加到要导入的ServiceDescriptions值的集合中。 ServiceDescriptionImporter sdi = new ServiceDescriptionImporter(); sdi.ProtocolName = "soap"; sdi.AddServiceDescription(sd, "", ""); // 4. 使用 CodeDom 编译客户端代理类。 //CodeNamespace表示命名空间声明 CodeNamespace cn = new CodeNamespace(@namespace);// 为代理类添加命名空间,缺省为全局空间。 logger.Info(cn.Name); //生成客户端代理类代码 // CodeCompileUnit会提供一个CodeDOM程式圆形的容器,CodeCompileUnit含有一个集合, //可以储存含有CodeDOM原始程式码原形,专案参考的组件集合以及专案组件属性集合的CodeNamespace物件。 CodeCompileUnit ccu = new CodeCompileUnit(); ccu.Namespaces.Add(cn); sdi.Import(cn, ccu); // CSharpCodeProvider类提供存取C#程式码产生器和程式码编译器的执行个体。 CodeDomProvider provider = new CSharpCodeProvider(); //设定编译参数 //创建编译器的参数实例 CompilerParameters cplist = new CompilerParameters(); cplist.GenerateExecutable = false; cplist.GenerateInMemory = true; cplist.ReferencedAssemblies.Add("System.dll"); cplist.ReferencedAssemblies.Add("System.XML.dll"); cplist.ReferencedAssemblies.Add("System.Web.Services.dll"); cplist.ReferencedAssemblies.Add("System.Data.dll"); //编译代理类 //CompilerResults表示从编译器返回的编译结果。使用指定的编译器设定, //根据CodeCompileUnit物件之指定阵列所包含的System.CodeDom树状结构,编译一个组件。 CompilerResults cr = provider.CompileAssemblyFromDom(cplist, ccu); //5. 使用 Reflection 调用 WebService 。 if (true == cr.Errors.HasErrors) { System.Text.StringBuilder sb = new System.Text.StringBuilder(); foreach (System.CodeDom.Compiler.CompilerError ce in cr.Errors) { sb.Append(ce.ToString()); sb.Append(System.Environment.NewLine); } logger.Error(sb.ToString()); //throw new Exception(sb.ToString()); } //生成代理实例,并调用方法 System.Reflection.Assembly assembly = cr.CompiledAssembly; // 如果在前面为代理类添加了命名空间,此处需要将命名空间添加到类型前面。 Type t = assembly.GetType(@namespace + "." + className, true, true); //Type t = assembly.GetType(className, true, true); // Activator类包含特定的方法,用以在本地或从远程创建对象类型,或获取对现有远程对象的引用。 //无法继承此类Activator.CreateInstance 方法 使用与指定参数匹配程度最高的构造函数创建指定类型的实例。 object obj = Activator.CreateInstance(t); //MethodInfo 的实例可以通过调用GetMethods或者Type对象或派生自Type的对象的GetMethod方法来获取, //还可以通过调用表示泛型方法定义的 MethodInfo 的MakeGenericMethod方法来获取。 System.Reflection.MethodInfo mi = t.GetMethod(methodName); logger.Info(mi.DeclaringType); return mi.Invoke(obj, args); } catch (Exception ex) { logger.Error(ex); //throw new Exception(ex.InnerException.Message, new Exception(ex.InnerException.StackTrace)); return null; } } private static string GetWsClassName(string wsUrl) { string[] parts = wsUrl.Split('/'); string[] pps = parts[parts.Length - 1].Split('.'); return pps[0]; } private static string GetNamespace(String URL) { HttpWebRequest request = (HttpWebRequest)WebRequest.Create(URL + "?WSDL"); SetWebRequest(request); WebResponse response = request.GetResponse(); StreamReader sr = new StreamReader(response.GetResponseStream(), Encoding.UTF8); XmlDocument doc = new XmlDocument(); doc.LoadXml(sr.ReadToEnd()); sr.Close(); string ns= doc.SelectSingleNode("//@targetNamespace").Value; return ns.Split('/')[ns.Split('/').Length - 1]; } private static void SetWebRequest(HttpWebRequest request) { request.Credentials = CredentialCache.DefaultCredentials; request.Timeout = 10000; } #endregion