动态调用webservices

2024-03-23 14:08
文章标签 动态 调用 webservices

本文主要是介绍动态调用webservices,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

通常我们在程序中需要调用WebService时,都是通过“添加Web引用”,让VS.NET环境来为我们生成服务代理,然后调用对应的Web服务。这样是使工作简单了,但是却和提供Web服务的URL、方法名、参数绑定在一起了,这是VS.NET自动为我们生成Web服务代理的限制。如果哪一天发布Web服务的URL改变了,则我们需要重新让VS.NET生成代理,并重新编译。在某些情况下,这可能是不能忍受的,我们需要动态调用WebService的能力。比如我们可以把Web服务的URL保存在配置文件中,这样,当服务URL改变时,只需要修改配置文件就可以了。
     说了这么多,实际上我们要实现这样的功能:

public static object InvokeWebService( string url,   string methodname, object [] args)

     其中,url是Web服务的地址,methodname是要调用服务方法名,args是要调用Web服务所需的参数,返回值就是web服务返回的结果了。

     要实现这样的功能,你需要这几个方面的技能:反射、CodeDom、编程使用C#编译器、WebService。在了解这些知识后,就可以容易的实现web服务的动态调用了:

         #region InvokeWebService
        
// 动态调用web服务
         public static object InvokeWebService( string url, string methodname, object [] args)
         {
            
return WebServiceHelper.InvokeWebService(url , null ,methodname ,args) ;
         }

        
public static object InvokeWebService( string url,   string classname, string methodname, object [] args)
         {
            
string @namespace = " EnterpriseServerBase.WebService.DynamicWebCalling " ;
            
if ((classname == null ) || (classname == "" ))
             {
                 classname
= WebServiceHelper.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();
                
                
// 设定编译参数
                 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 = 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)
             {
                
throw new Exception(ex.InnerException.Message, new Exception(ex.InnerException.StackTrace));
             }
         }

        
private static string GetWsClassName( string wsUrl)
         {
            
string [] parts = wsUrl.Split( ' / ' ) ;
            
string [] pps    = parts[parts.Length - 1 ].Split( ' . ' ) ;

            
return pps[ 0 ] ;
         }
        
#endregion


     上面的注释已经很好的说明了各代码段的功能,下面给个例子看看,这个例子是通过访问http://www.webservicex.net/globalweather.asmx 服务来获取各大城市的天气状况。

             string url = " http://www.webservicex.net/globalweather.asmx " ;
            
string [] args = new string [ 2 ] ;
             args[
0 ] = this .textBox_CityName.Text ;
             args[
1 ] = " China " ;
            
object result = WebServiceHelper.InvokeWebService(url , " GetWeather " ,args) ;
            
this .label_Result.Text = result.ToString() ;


     上述的例子中,调用web服务使用了两个参数,第一个是城市的名字,第二个是国家的名字,Web服务返回的是XML文档,可以从其中解析出温度、风力等天气情况。
    
     最后说一下,C#虽然仍属于静态语言之列,但是其动态能力也是很强大的,不信,你可以看看Spring.net的AOP实现,这种“无侵入”的AOP实现比通常的.NET声明式AOP实现(一般是通过AOP Attribute)要漂亮的多。

这篇关于动态调用webservices的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/838533

相关文章

MySQL中动态生成SQL语句去掉所有字段的空格的操作方法

《MySQL中动态生成SQL语句去掉所有字段的空格的操作方法》在数据库管理过程中,我们常常会遇到需要对表中字段进行清洗和整理的情况,本文将详细介绍如何在MySQL中动态生成SQL语句来去掉所有字段的空... 目录在mysql中动态生成SQL语句去掉所有字段的空格准备工作原理分析动态生成SQL语句在MySQL

Java调用C++动态库超详细步骤讲解(附源码)

《Java调用C++动态库超详细步骤讲解(附源码)》C语言因其高效和接近硬件的特性,时常会被用在性能要求较高或者需要直接操作硬件的场合,:本文主要介绍Java调用C++动态库的相关资料,文中通过代... 目录一、直接调用C++库第一步:动态库生成(vs2017+qt5.12.10)第二步:Java调用C++

C#如何动态创建Label,及动态label事件

《C#如何动态创建Label,及动态label事件》:本文主要介绍C#如何动态创建Label,及动态label事件,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C#如何动态创建Label,及动态label事件第一点:switch中的生成我们的label事件接着,

SpringCloud动态配置注解@RefreshScope与@Component的深度解析

《SpringCloud动态配置注解@RefreshScope与@Component的深度解析》在现代微服务架构中,动态配置管理是一个关键需求,本文将为大家介绍SpringCloud中相关的注解@Re... 目录引言1. @RefreshScope 的作用与原理1.1 什么是 @RefreshScope1.

MyBatis 动态 SQL 优化之标签的实战与技巧(常见用法)

《MyBatis动态SQL优化之标签的实战与技巧(常见用法)》本文通过详细的示例和实际应用场景,介绍了如何有效利用这些标签来优化MyBatis配置,提升开发效率,确保SQL的高效执行和安全性,感... 目录动态SQL详解一、动态SQL的核心概念1.1 什么是动态SQL?1.2 动态SQL的优点1.3 动态S

在C#中调用Python代码的两种实现方式

《在C#中调用Python代码的两种实现方式》:本文主要介绍在C#中调用Python代码的两种实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C#调用python代码的方式1. 使用 Python.NET2. 使用外部进程调用 Python 脚本总结C#调

SpringCloud之LoadBalancer负载均衡服务调用过程

《SpringCloud之LoadBalancer负载均衡服务调用过程》:本文主要介绍SpringCloud之LoadBalancer负载均衡服务调用过程,具有很好的参考价值,希望对大家有所帮助,... 目录前言一、LoadBalancer是什么?二、使用步骤1、启动consul2、客户端加入依赖3、以服务

Vue 调用摄像头扫描条码功能实现代码

《Vue调用摄像头扫描条码功能实现代码》本文介绍了如何使用Vue.js和jsQR库来实现调用摄像头并扫描条码的功能,通过安装依赖、获取摄像头视频流、解析条码等步骤,实现了从开始扫描到停止扫描的完整流... 目录实现步骤:代码实现1. 安装依赖2. vue 页面代码功能说明注意事项以下是一个基于 Vue.js

mybatis-plus 实现查询表名动态修改的示例代码

《mybatis-plus实现查询表名动态修改的示例代码》通过MyBatis-Plus实现表名的动态替换,根据配置或入参选择不同的表,本文主要介绍了mybatis-plus实现查询表名动态修改的示... 目录实现数据库初始化依赖包配置读取类设置 myBATis-plus 插件测试通过 mybatis-plu

讯飞webapi语音识别接口调用示例代码(python)

《讯飞webapi语音识别接口调用示例代码(python)》:本文主要介绍如何使用Python3调用讯飞WebAPI语音识别接口,重点解决了在处理语音识别结果时判断是否为最后一帧的问题,通过运行代... 目录前言一、环境二、引入库三、代码实例四、运行结果五、总结前言基于python3 讯飞webAPI语音