动态调用WebService的两种方法(多线程)
发布时间:2020-12-16 22:44:13 所属栏目:安全 来源:网络整理
导读:? ? ? ? ? ? ? ?在.net中,可以添加Web 引用来添加WebService,但是这种方法的缺陷是当WebService内的方法一变动,引用的系统这边就必须更新引用,重新编译,再发布,是不是很麻烦?也未可预知? ? ? ? ? ? ? ? ?那么就使用动态调用WebService吧! 第1种,具
? ? ? ? ? ? ? ?在.net中,可以添加Web 引用来添加WebService,但是这种方法的缺陷是当WebService内的方法一变动,引用的系统这边就必须更新引用,重新编译,再发布,是不是很麻烦?也未可预知? ? ? ? ? ? ? ? ?那么就使用动态调用WebService吧! 第1种,具体步骤: 代码如下: /// <summary> /// 在.Net环境 下,最常用的方法就是采用代理类来调用WebService,可以通过改变代理类的Url属性来实现动态调用, /// 但当xmlns改变时就会出错,似乎要重新 绑定Webservice并重新编译后才能再次运行。 /// 此法子是一种动态编译并动态调用WebService的方式,这种方法效率低,而且需要有较高 的权限,否则编译失败。 /// 此法子的缺陷。。。。。都是泪啊。。。多线程下运行第二次就报错:无法从命名空间“****”导入绑定“SyncToContractDataCenterSoap” /// </summary> public class WebServiceHelper2 { #region InvokeWebService //动态调用web服务 public object InvokeWebService(string url,string methodname,object[] args) { return InvokeWebService(url,null,methodname,args); } public object InvokeWebService(string url,string classname,object[] args) { string @namespace = "TransactionDataSync.Common"; if ((classname == null) || (classname == "")) { classname = GetWsClassName(url); } try { //获取WSDL WebClient wc = new WebClient(); Stream stream = wc.OpenRead(url + "?wsdl"); ServiceDescription sd = ServiceDescription.Read(stream); ServiceDescriptionImporter sdi = new ServiceDescriptionImporter(); sdi.AddServiceDescription(sd,"",""); CodeNamespace cn = new CodeNamespace(@namespace); //生成客户端代理类代码 CodeCompileUnit ccu = new CodeCompileUnit(); ccu.Namespaces.Add(cn); sdi.Import(cn,ccu); // CSharpCodeProvider csc = new CSharpCodeProvider(); // ICodeCompiler icc = csc.CreateCompiler(); CodeDomProvider provider = CodeDomProvider.CreateProvider("CSharp"); //设定编译参数 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 cr = provider.CompileAssemblyFromDom(cplist,ccu); // CompilerResults cr = icc.CompileAssemblyFromDom(cplist,ccu); 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); } throw new Exception(sb.ToString()); } //生成代理实例,并调用方法 System.Reflection.Assembly assembly = cr.CompiledAssembly; Type t = assembly.GetType(@namespace + "." + classname,true,true); object obj = Activator.CreateInstance(t); System.Reflection.MethodInfo mi = t.GetMethod(methodname); return mi.Invoke(obj,args); } catch (Exception ex) { Logger.Error(string.Format("{0} {1} {2} {3}",url,ex.Message,ex.StackTrace)); throw new Exception(ex.InnerException.Message,new Exception(ex.InnerException.StackTrace)); } } public object CreateWebServiceInstance(string url,string packageName,out Type t) { return CreateWebServiceInstance(url,packageName,out t); } public object CreateWebServiceInstance(string url,out Type t) { string @namespace = "TransactionDataSync.Common."+packageName; if ((classname == null) || (classname == "")) { classname = GetWsClassName(url); } Stream stream = null; try { //获取WSDL WebClient wc = new WebClient(); stream = wc.OpenRead(url + "?WSDL"); ServiceDescription sd = ServiceDescription.Read(stream); ServiceDescriptionImporter sdi = new ServiceDescriptionImporter(); sdi.AddServiceDescription(sd,""); CodeNamespace cn = new CodeNamespace(@namespace); //生成客户端代理类代码 CodeCompileUnit ccu = new CodeCompileUnit(); ccu.Namespaces.Add(cn); sdi.Import(cn,ccu); // CSharpCodeProvider csc = new CSharpCodeProvider(); // ICodeCompiler icc = csc.CreateCompiler(); CodeDomProvider provider = CodeDomProvider.CreateProvider("CSharp"); //设定编译参数 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 cr = provider.CompileAssemblyFromDom(cplist,ccu); // CompilerResults cr = icc.CompileAssemblyFromDom(cplist,ccu); 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); } throw new Exception(sb.ToString()); } //生成代理实例 System.Reflection.Assembly assembly = cr.CompiledAssembly; t = assembly.GetType(@namespace + "." + classname,true); object obj = Activator.CreateInstance(t); return obj; } catch (Exception ex) { Logger.Error(string.Format("{0} {1} {2}",ex.StackTrace)); if(stream!=null) { stream.Close(); stream.Dispose(); } throw ex; } } public object InvokeMethod(object obj,Type t,object[] args) { try { System.Reflection.MethodInfo mi = t.GetMethod(methodname); return mi.Invoke(obj,args); } catch(Exception ex) { Logger.Error(string.Format("{0} {1} {2}",ex.StackTrace)); throw ex; } } private string GetWsClassName(string wsUrl) { string[] parts = wsUrl.Split('/'); string[] pps = parts[parts.Length - 1].Split('.'); return pps[0]; } #endregion
调用时的代码:
object instance = WebServiceHelper2.CreateWebServiceInstance(addr.Url,name,out t); DataSet dsContract = WebServiceHelper2.InvokeMethod(instance,t,"GetContractData",new object[]{authorizationCode}) as DataSet; 缺陷:
? ? ? ? ? ? ? ?第2种动态调用的法子:利用WebRequest/WebResponse进行WebService调用的类
? ? ? ? ? ? ? ? 动态调用的类如下:
? ? ? ?
using System; using System.Collections; using System.Collections.Generic; using System.Data; using System.IO; using System.Linq; using System.Net; using System.Text; using System.Xml; using System.Xml.Serialization; namespace TransactionDataSync.Common { /// <summary> /// 利用WebRequest/WebResponse进行WebService调用的类 /// </summary> /// <remarks> /// 作成者:cyf /// </remarks> public class WebServiceCaller { //缓存xmlNamespace,避免重复调用GetNamespace private static Hashtable XML_NAMESPACE = new Hashtable(); /// <summary> /// 通过SOAP协议动态调用webservice /// </summary> /// <param name="url"> webservice地址</param> /// <param name="methodName"> 调用方法名</param> /// <param name="pars"> 参数表</param> /// <returns> 结果集xml</returns> public static XmlDocument QuerySoapWebService(String url,String methodName,Hashtable pars) { if (XML_NAMESPACE.ContainsKey(url)) { // 名字空间在缓存中存在时,读取缓存,然后执行调用 return QuerySoapWebService(url,methodName,pars,XML_NAMESPACE[url].ToString()); } else { // 名字空间不存在时直接从wsdl的请求中读取名字空间,然后执行调用 return QuerySoapWebService(url,GetNamespace(url)); } } /// <summary> /// 通过SOAP协议动态调用webservice /// </summary> /// <param name="url"> webservice地址</param> /// <param name="methodName"> 调用方法名</param> /// <param name="pars"> 参数表</param> /// <param name="xmlNs"> 名字空间</param> /// <returns> 结果集</returns> private static XmlDocument QuerySoapWebService(String url,Hashtable pars,string xmlNs) { XML_NAMESPACE[url] = xmlNs;//加入缓存,提高效率 // 获取请求对象 HttpWebRequest request = (HttpWebRequest)HttpWebRequest.Create(url); // 设置请求head request.Method = "POST"; request.ContentType = "text/xml; charset=utf-8"; request.Headers.Add("SOAPAction",""" + xmlNs + (xmlNs.EndsWith("/") ? "" : "/") + methodName + """); // 设置请求身份 SetWebRequest(request); // 获取soap协议 byte[] data = EncodeParsToSoap(pars,xmlNs,methodName); // 将soap协议写入请求 WriteRequestData(request,data); XmlDocument returnDoc = new XmlDocument(); XmlDocument returnValueDoc = new XmlDocument(); // 读取服务端响应 returnDoc = ReadXmlResponse(request.GetResponse()); XmlNamespaceManager mgr = new XmlNamespaceManager(returnDoc.NameTable); mgr.AddNamespace("soap","http://schemas.xmlsoap.org/soap/envelope/"); // 返回结果 string RetXml= returnDoc.SelectSingleNode("//soap:Body/*/*",mgr).InnerXml; returnValueDoc.LoadXml("<root>" + RetXml + "</root>"); AddDelaration(returnValueDoc); /* System.Data.DataSet ds = new System.Data.DataSet(); XmlNodeReader reader = new XmlNodeReader(returnValueDoc); ds.ReadXml(reader);*/ // return returnValueDoc.OuterXml; return returnValueDoc; } /// <summary> /// 获取wsdl中的名字空间 /// </summary> /// <param name="url"> wsdl地址</param> /// <returns> 名字空间</returns> private static string GetNamespace(String url) { // 创建wsdl请求对象,并从中读取名字空间 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(); return doc.SelectSingleNode("//@targetNamespace").Value; } /// <summary> /// 加入soapheader节点 /// </summary> /// <param name="doc"> soap文档</param> private static void InitSoapHeader(XmlDocument doc) { // 添加soapheader节点 XmlElement soapHeader = doc.CreateElement("soap","Header","http://schemas.xmlsoap.org/soap/envelope/"); //XmlElement soapId = doc.CreateElement("userid"); //soapId.InnerText = ID; //XmlElement soapPwd = doc.CreateElement("userpwd"); //soapPwd.InnerText = PWD; //soapHeader.AppendChild(soapId); //soapHeader.AppendChild(soapPwd); doc.ChildNodes[0].AppendChild(soapHeader); } /// <summary> /// 将以字节数组的形式返回soap协议 /// </summary> /// <param name="pars"> 参数表</param> /// <param name="xmlNs"> 名字空间</param> /// <param name="methodName"> 方法名</param> /// <returns> 字节数组</returns> private static byte[] EncodeParsToSoap(Hashtable pars,String xmlNs,String methodName) { XmlDocument doc = new XmlDocument(); // 构建soap文档 doc.LoadXml("<soap:Envelope xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance/" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/"></soap:Envelope>"); // 加入soapbody节点 InitSoapHeader(doc); // 创建soapbody节点 XmlElement soapBody = doc.CreateElement("soap","Body","http://schemas.xmlsoap.org/soap/envelope/"); // 根据要调用的方法创建一个方法节点 XmlElement soapMethod = doc.CreateElement(methodName); soapMethod.SetAttribute("xmlns",xmlNs); // 遍历参数表中的参数键 foreach (string key in pars.Keys) { // 根据参数表中的键值对,生成一个参数节点,并加入方法节点内 XmlElement soapPar = doc.CreateElement(key); soapPar.InnerXml = ObjectToSoapXml(pars[key]); soapMethod.AppendChild(soapPar); } // soapbody节点中加入方法节点 soapBody.AppendChild(soapMethod); // soap文档中加入soapbody节点 doc.DocumentElement.AppendChild(soapBody); // 添加声明 AddDelaration(doc); // 传入的参数有DataSet类型,必须在序列化后的XML中的diffgr:diffgram/NewDataSet节点加xmlns='' 否则无法取到每行的记录。 XmlNode node = doc.DocumentElement.SelectSingleNode("//NewDataSet"); if (node != null) { XmlAttribute attr = doc.CreateAttribute("xmlns"); attr.InnerText = ""; node.Attributes.Append(attr); } // 以字节数组的形式返回soap文档 return Encoding.UTF8.GetBytes(doc.OuterXml); } /// <summary> /// 将参数对象中的内容取出 /// </summary> /// <param name="o">参数值对象</param> /// <returns>字符型值对象</returns> private static string ObjectToSoapXml(object o) { XmlSerializer mySerializer = new XmlSerializer(o.GetType()); MemoryStream ms = new MemoryStream(); mySerializer.Serialize(ms,o); XmlDocument doc = new XmlDocument(); doc.LoadXml(Encoding.UTF8.GetString(ms.ToArray())); if (doc.DocumentElement != null) { return doc.DocumentElement.InnerXml; } else { return o.ToString(); } } /// <summary> /// 设置请求身份 /// </summary> /// <param name="request"> 请求</param> private static void SetWebRequest(HttpWebRequest request) { request.Credentials = CredentialCache.DefaultCredentials; //request.Timeout = 10000; } /// <summary> /// 将soap协议写入请求 /// </summary> /// <param name="request"> 请求</param> /// <param name="data"> soap协议</param> private static void WriteRequestData(HttpWebRequest request,byte[] data) { request.ContentLength = data.Length; Stream writer = request.GetRequestStream(); writer.Write(data,data.Length); writer.Close(); } /// <summary> /// 将响应对象读取为xml对象 /// </summary> /// <param name="response"> 响应对象</param> /// <returns> xml对象</returns> private static XmlDocument ReadXmlResponse(WebResponse response) { StreamReader sr = new StreamReader(response.GetResponseStream(),Encoding.UTF8); String retXml = sr.ReadToEnd(); sr.Close(); XmlDocument doc = new XmlDocument(); doc.LoadXml(retXml); return doc; } /// <summary> /// 给xml文档添加声明 /// </summary> /// <param name="doc"> xml文档</param> private static void AddDelaration(XmlDocument doc) { XmlDeclaration decl = doc.CreateXmlDeclaration("1.0","utf-8",null); doc.InsertBefore(decl,doc.DocumentElement); } public static String QuerySoapWebServiceString(String url,Hashtable pars) { XmlDocument doc = QuerySoapWebService(url,pars); return doc.InnerText; } public static DataSet QuerySoapWebServiceDataSet(String url,pars); System.Data.DataSet ds = new System.Data.DataSet(); using (XmlNodeReader reader = new XmlNodeReader(doc)) { ds.ReadXml(reader); } return ds; } } }
使用时的调用如下:
Hashtable htParms = new Hashtable(); htParms.Add("authorizationCode",authCode); DataSet dsComm = WebServiceCaller.QuerySoapWebServiceDataSet(url,"GetReceivedCommissionData",htParms); 或者 DataSet dsSelected = new DataSet(); dsSelected.Tables.Add(dtSelected); //需要返回给业务系统的数据集,表示哪些数据已收集 Hashtable htParms = new Hashtable(); htParms.Add("authorizationCode",addr.AuthorizationCode); htParms.Add("dsData",dtSelected); string result = WebServiceCaller.QuerySoapWebServiceString(url,htParms);//传入的参数是一个String类型和一个DataSet类型 动态调用时用的是Soap协议的方式来请求的,而不是Http Post,因为当WebService中的方法传入参数中有DataSet类型,将无法用Http Post方式请求,只能用Soap。
在多线程中,同时调多个WebService,此种方法不会报错。因此我选择了它。。。。
(编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |