(ZT)动态调用web服务

通常我们在程序中需要调用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)要漂亮的多。

使用LabVIEW调用ZT600打印机的过程包括以下几个步骤: 1. 安装驱动程序:首先,需要将Zebra打印机的驱动程序安装到计算机上。可以从Zebra官方网站下载最新的驱动程序,并按照安装向导进行安装。 2. 创建VI:在LabVIEW中创建一个新的虚拟仪器(VI),作为与ZT600打印机通信的界面。可以通过LabVIEW的开发环境进行创建,选择适当的界面元素和功能。 3. 配置串口通信:使用LabVIEW的串口通信功能,将计算机与打印机进行连接。在VI中添加串口通信的功能模块,并设置正确的串口参数,如波特率、数据位、校验位等。 4. 编写命令脚本:通过串口发送命令来控制打印机的功能。在LabVIEW中,可以使用串口通信模块发送特定的命令字符串,来实现打印机的各种功能,如打印文本、条码、图像等。 5. 打印文件:通过LabVIEW将需要打印的文件传输给打印机。可以使用LabVIEW的文件处理功能,读取需要打印的文件,并将其转换为适当的格式,然后通过串口发送给打印机。 6. 错误处理:在LabVIEW的VI中添加适当的错误处理机制,以捕捉和处理可能出现的错误情况。可以使用LabVIEW的错误处理功能模块,以及条件判断和循环结构来实现错误处理机制。 7. 调试和验证:在完成上述步骤后,使用实际的ZT600打印机进行调试和验证。可以通过连接真实的打印机,并运行LabVIEW中的VI,来验证打印机是否正常工作,并输出预期的打印结果。 通过以上步骤,可以在LabVIEW中成功对ZT600打印机进行调用,并实现所需的打印功能。注意确保电脑与打印机正确连接,并且配置正确的串口参数。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值