C# 调用WebService的方法

 更新时间:2020年08月03日 15:29:11   作者:osc_omyprm56  
这篇文章主要介绍了C# 调用WebService的方法,文中讲解非常细致,代码帮助大家更好的理解和学习,感兴趣的朋友可以了解下

一、前言

在日常工作中,如果涉及到与第三方进行接口对接,有的会使用WebService的方式,这篇文章主要讲解在.NET Framework中如何调用WebService。首先我们创建一个WebService,里面有两个方法:一个无参的方法,一个有参的方法:

创建好了WebService以后,把WebService部署到IIS上,并确保可以访问

二、静态引用

这种方式是通过添加静态引用的方式调用WebService。首先创建一个Winform程序,界面上有一个按钮,点击按钮调用WebService:

然后添加静态引用。在要调用WebService的项目上选择引用,然后右键选择“添加服务引用”,如下图所示:

然后输入IIS上部署的WebService地址:

最后点击“确定”按钮即可完成静态引用WebService,添加完成以后的项目结构如下图所示:

添加完引用以后,就可以编写代码了:

/// <summary>
/// 静态调用WebService
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void btn_Static_Click(object sender, EventArgs e)
{
  // 实例化类
  CallWebService.TestWebSoapClient client = new CallWebService.TestWebSoapClient();
  // 调用无参的HelloWorld方法
  string value1= client.HelloWorld();
  // 调用有参的方法
  string value2 = client.Test("有参方法");
  // 输出
  MessageBox.Show($"无参方法返回值:{value1},有参方法返回值:{value2}");
}

运行程序测试:

这样就可以实现调用WebService了。

三、动态调用

上面我们说了如何使用静态引用的方式调用WebService,但是这种方式有一个缺点:如果发布的WebService地址改变,那么就要重新添加WebService的引用。如果是现有的WebService发生了改变,也要更新现有的服务引用,这需要把代码放到现场才可以。那么有没有什么方式可以解决这种问题呢?那就是使用动态调用WebService的方法。

我们在配置文件里面添加配置,把WebService的地址、WebService提供的类名、要调用的方法名称,都写在配置文件里面:

<appSettings>
  <!--WebService地址-->
  <add key="WebServiceAddress" value="http://localhost:9008/TestWeb.asmx"/>
  <!--WebService提供的类名-->
  <add key="ClassName" value="TestWeb"/>
  <!--WebService方法名-->
  <add key="MethodName" value="Test"/>
  <!--存放dll文件的地址-->
  <add key="FilePath" value="E:\Test"/>
</appSettings>

在界面上添加一个按钮,点击按钮可以动态调用WebService,新建一个帮助类:

using System;
using System.CodeDom;
using System.CodeDom.Compiler;
using System.IO;
using System.Net;
using System.Text;
using System.Web;
using System.Web.Caching;
using System.Web.Services.Description;
using System.Xml.Serialization;

namespace WebServiceDemo
{
  public class WebServiceHelper
  {
    /// <summary>
    /// 生成dll文件保存到本地
    /// </summary>
    /// <param name="url">WebService地址</param>
    /// <param name="className">类名</param>
    /// <param name="methodName">方法名</param>
    /// <param name="filePath">保存dll文件的路径</param>
    public static void CreateWebServiceDLL(string url,string className, string methodName,string filePath )
    {
      // 1. 使用 WebClient 下载 WSDL 信息。
      WebClient web = new WebClient();
      Stream stream = web.OpenRead(url + "?WSDL");
      // 2. 创建和格式化 WSDL 文档。
      ServiceDescription description = ServiceDescription.Read(stream);
      //如果不存在就创建file文件夹
      if (Directory.Exists(filePath) == false)
      {
        Directory.CreateDirectory(filePath);
      }

      if (File.Exists(filePath + className + "_" + methodName + ".dll"))
      {
        //判断缓存是否过期
        var cachevalue = HttpRuntime.Cache.Get(className + "_" + methodName);
        if (cachevalue == null)
        {
          //缓存过期删除dll
          File.Delete(filePath + className + "_" + methodName + ".dll");
        }
        else
        {
          // 如果缓存没有过期直接返回
          return;
        }
      }

      // 3. 创建客户端代理代理类。
      ServiceDescriptionImporter importer = new ServiceDescriptionImporter();
      // 指定访问协议。
      importer.ProtocolName = "Soap";
      // 生成客户端代理。
      importer.Style = ServiceDescriptionImportStyle.Client; 
      importer.CodeGenerationOptions = CodeGenerationOptions.GenerateProperties | CodeGenerationOptions.GenerateNewAsync;
      // 添加 WSDL 文档。
      importer.AddServiceDescription(description, null, null);
      // 4. 使用 CodeDom 编译客户端代理类。
      // 为代理类添加命名空间,缺省为全局空间。
      CodeNamespace nmspace = new CodeNamespace();    
      CodeCompileUnit unit = new CodeCompileUnit();
      unit.Namespaces.Add(nmspace);
      ServiceDescriptionImportWarnings warning = importer.Import(nmspace, unit);
      CodeDomProvider provider = CodeDomProvider.CreateProvider("CSharp");
      CompilerParameters parameter = new CompilerParameters();
      parameter.GenerateExecutable = false;
      // 可以指定你所需的任何文件名。
      parameter.OutputAssembly = filePath + className + "_" + methodName + ".dll"; 
      parameter.ReferencedAssemblies.Add("System.dll");
      parameter.ReferencedAssemblies.Add("System.XML.dll");
      parameter.ReferencedAssemblies.Add("System.Web.Services.dll");
      parameter.ReferencedAssemblies.Add("System.Data.dll");
      // 生成dll文件,并会把WebService信息写入到dll里面
      CompilerResults result = provider.CompileAssemblyFromDom(parameter, unit);
      if (result.Errors.HasErrors)
      {
        // 显示编译错误信息
        System.Text.StringBuilder sb = new StringBuilder();
        foreach (CompilerError ce in result.Errors)
        {
          sb.Append(ce.ToString());
          sb.Append(System.Environment.NewLine);
        }
        throw new Exception(sb.ToString());
      }
      //记录缓存
      var objCache = HttpRuntime.Cache;
      // 缓存信息写入dll文件
      objCache.Insert(className + "_" + methodName, "1", null, DateTime.Now.AddMinutes(5), TimeSpan.Zero, CacheItemPriority.High, null);
    }
  }
}

动态调用WebService代码:

/// <summary>
/// 动态调用WebService
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void btn_Dynamic_Click(object sender, EventArgs e)
{
  // 读取配置文件,获取配置信息
  string url = ConfigurationManager.AppSettings["WebServiceAddress"];
  string className = ConfigurationManager.AppSettings["ClassName"];
  string methodName = ConfigurationManager.AppSettings["MethodName"];
  string filePath = ConfigurationManager.AppSettings["FilePath"];
  // 调用WebServiceHelper
  WebServiceHelper.CreateWebServiceDLL(url, className, methodName, filePath);
  // 读取dll内容
  byte[] filedata = File.ReadAllBytes(filePath + className + "_" + methodName + ".dll");
  // 加载程序集信息
  Assembly asm = Assembly.Load(filedata);
  Type t = asm.GetType(className);
  // 创建实例
  object o = Activator.CreateInstance(t);
  MethodInfo method = t.GetMethod(methodName);
  // 参数
  object[] args = {"动态调用WebService" };
  // 调用访问,获取方法返回值
  string value = method.Invoke(o, args).ToString();
  //输出返回值
  MessageBox.Show($"返回值:{value}");
}

程序运行结果:

如果说类名没有提供,可以根据url来自动获取类名:

/// <summary>
/// 根据WebService的url地址获取className
/// </summary>
/// <param name="wsUrl">WebService的url地址</param>
/// <returns></returns>
private string GetWsClassName(string wsUrl)
{
  string[] parts = wsUrl.Split('/');
  string[] pps = parts[parts.Length - 1].Split('.');
  return pps[0];
}

以上就是C# 调用WebService的方法的详细内容,更多关于C# 调用WebService的资料请关注脚本之家其它相关文章!

相关文章

  • 详解C#应用程序如何实现多屏显示

    详解C#应用程序如何实现多屏显示

    这篇文章主要为大家详细介绍了C#如何实现把主屏运行程序中多个窗体移动到各个扩展屏幕位置显示,感兴趣的小伙伴可以跟随小编一起学习一下
    2024-05-05
  • C#设计模式之职责链模式示例详解

    C#设计模式之职责链模式示例详解

    这篇文章主要给大家介绍了关于C#设计模式之职责链模式的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-12-12
  • DevExpress之ChartControl实现柱状图演示实例

    DevExpress之ChartControl实现柱状图演示实例

    这篇文章主要介绍了DevExpress中ChartControl实现柱状图演示方法,实例展示了相关绘图函数的具体用法,具有一定的实用价值,需要的朋友可以参考下
    2014-10-10
  • C# Onnx实现轻量实时的M-LSD直线检测

    C# Onnx实现轻量实时的M-LSD直线检测

    这篇文章主要为大家详细介绍了C#如何结合Onnx实现轻量实时的M-LSD直线检测,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2023-11-11
  • 杂谈try-catch-finally异常处理

    杂谈try-catch-finally异常处理

    这篇文章主要介绍了杂谈try-catch-finally异常处理的相关资料,需要的朋友可以参考下
    2016-01-01
  • C# 异步多线程入门到精通之ThreadPool篇

    C# 异步多线程入门到精通之ThreadPool篇

    ThreadPool 是 .net 2.0 时代的产物,有了 Thread 为什么还会有 ThreadPool 呢?ThreadPool 可以做到限制线程数量、重用线程
    2021-11-11
  • C#使用PuppeteerSharp库的示例代码

    C#使用PuppeteerSharp库的示例代码

    PuppeteerSharp 是一个用于在 .NET 平台上控制无头(Headless) Chrome 浏览器的库,本文主要为大家详细介绍了C#中使用PuppeteerSharp库的工具类,需要的可以了解下
    2023-12-12
  • c# 实现控件(ocx)中的事件详解

    c# 实现控件(ocx)中的事件详解

    这篇文章主要介绍了c# 实现控件(ocx)中的事件详解,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-12-12
  • C#中Invoke的用法讲解

    C#中Invoke的用法讲解

    这篇文章主要介绍了C#中Invoke的用法讲解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-07-07
  • C#使用LINQ查询文件列表并找出最大文件

    C#使用LINQ查询文件列表并找出最大文件

    在现代 C# 开发中,LINQ (Language Integrated Query) 提供了一种强大而优雅的方式来处理集合数据,本文将详细介绍如何使用 LINQ 查询文件系统中的文件,并找出最大的文件数量,需要的朋友可以参考下
    2024-10-10

最新评论