C#调用webservice接口的最新方法教程

时间:2021-09-09 07:22:28

前言

web service也叫xml web service webservice是一种可以接收从internet或者intranet上的其它系统中传递过来的请求,轻量级的独立的通讯技术。是:通过soap在web上提供的软件服务,使用wsdl文件进行说明,并通过uddi进行注册。

xml:(extensible markup language)扩展型可标记语言。面向短期的临时数据处理、面向万维网络,是soap的基础。

soap:(simple object access protocol)简单对象存取协议。是xml web service 的通信协议。当用户通过uddi找到你的wsdl描述文档后,他通过可以soap调用你建立的web服务中的一个或多个操作。soap是xml文档形式的调用方法的规范,它可以支持不同的底层接口,像http(s)或者smtp。

wsdl:(web services description language) wsdl 文件是一个 xml 文档,用于说明一组 soap 消息以及如何交换这些消息。大多数情况下由软件自动生成和使用。

uddi (universal description, discovery, and integration) 是一个主要针对web服务供应商和使用者的新项目。在用户能够调用web服务之前,必须确定这个服务内包含哪些商务方法,找到被调用的接口定义,还要在服务端来编制软件,uddi是一种根据描述文档来引导系统查找相应服务的机制。uddi利用soap消息机制(标准的xml/http)来发布,编辑,浏览以及查找注册信息。它采用xml格式来封装各种不同类型的数据,并且发送到注册中心或者由注册中心来返回需要的数据。

c#调用webservice接口

我们在日常开发中,经常遇到c#调用webservice的情况,通常来说如果webservice是用vs+c#来开发的,问题一般来说不大,直接web引用,然后调用就ok了。

流程如下:

C#调用webservice接口的最新方法教程

C#调用webservice接口的最新方法教程

C#调用webservice接口的最新方法教程

C#调用webservice接口的最新方法教程

下面就是进行调用,就这么简单。

C#调用webservice接口的最新方法教程

但如果webservice是用java或者其它语言或者其它工具生成的话,使用vs+c#来调用就经常遇到问题;就是使用上面的方法显得很不好使,经常是使用soap ui调用没有问题,但使用上面的方法却调用报错,经常是500的错误。当你联系webservice提供商时通常会说soap ui都能调用得到,你们用代码为啥子调用不到,问题出在你们的调用方法上。

在我们向其它公司提供webservice的时候,经常也会出现这样的问题,以前我们一直都以为soap ui能够调用,那么代码也就一定能够调用得通,但经过实践,我们自己写demo调用自己的webservice时才发现,并不是别人的调用代码写的有问题,因为我们自己也无法将自己写的webservice调用得通,或者说没有找到正确的方法调用得通。

这时我们就要思考是否是soap ui能够调用得通的webservice就代码一定调用没有问题呢?或者说soap ui调用webservice和代码调用webservice的原理区别到底在哪里呢?

总结一下:

(1)soap ui能够调用成功,代码不一定能够调用成功,代码调用成功并且得到返回结果的前提是webservice可以按标准返回结果,但soap ui是只要按信封返回就可以收到结果而不管结果是否标准;

(2)如果webservice的header有用户名和密码的校验,使用soap ui可以调用成功并且得到返回结果,但使用上面web引用的方式却不行。对于这种情况,有以下方法可以调用成功:

重复上面web引用的方式,另外引用microsoft.web.services3(这个dll可以在这里下载)。

C#调用webservice接口的最新方法教程

手工修改reference.cs

C#调用webservice接口的最新方法教程

C#调用webservice接口的最新方法教程

以下是调用方法,重点在传入用户名和密码,修改继承类的重点也在于此;

C#调用webservice接口的最新方法教程

引用完成,调用就会成功了,结果如下:

C#调用webservice接口的最新方法教程

从上面可以看出web引用真的是简单粗暴,基本不用写几句代码就可以搞定绝大部分的webservice调用。

但是web引用却有在现实开发中常遇到的缺点:

(1)需要开发环境可以访问到的wsdl地址;

(2)如果webservice有变化需要更新web引用;

C#调用webservice接口的最新方法教程

那有没有其它方法呢?答案是有的。一种是直接封装信封,使用webrequest,代码如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
private void button2_click(object sender, eventargs e)
 {
  stringbuilder soap = new stringbuilder();
  soap.append("<soap:envelope xmlns:soap=\"http://www.w3.org/2003/05/soap-envelope\" xmlns:zls=\"www.zlsoft.cn\">");
  soap.append("<soap:header>");
  soap.append("<wsse:security soap:mustunderstand=\"true\" xmlns:wsse=\"http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-secext-1.0.xsd\" xmlns:wsu=\"http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-utility-1.0.xsd\">");
  soap.append("<wsse:usernametoken wsu:id=\"usernametoken-4\">");
  soap.append("<wsse:username>hjq</wsse:username>");//用户名
  soap.append("<wsse:password type=\"http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-username-token-profile-1.0#passwordtext\">123</wsse:password>");//口令
  soap.append("<wsse:nonce encodingtype=\"http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-soap-message-security-1.0#base64binary\">gc6ddgkjxo0iarc5kpqu3w==</wsse:nonce>");
  soap.append("<wsu:created>2017-11-01t05:11:22.805z</wsu:created>");
  soap.append("</wsse:usernametoken>");
  soap.append("</wsse:security>");
  soap.append("</soap:header>");
  soap.append("<soap:body>");
  soap.append("<zls:getuserinfo>");
  soap.append("<zls:appsys_id>trwetre</zls:appsys_id>");
  soap.append("<zls:appsys_token>sdafdsf</zls:appsys_token>");
  soap.append("<zls:sid>fdsafds</zls:sid>");
  soap.append("</zls:getuserinfo>");
  soap.append("</soap:body>");
  soap.append("</soap:envelope>");
 
  string url = "http://127.0.0.1:6998/services/hipservice?wsdl";
  var result = getsoapresource(url, soap.tostring());
 }
 
 
  public static string getsoapresource(string url, string datastr)
  {
  try
  {
   //request
   uri uri = new uri(url);
   webrequest webrequest = webrequest.create(uri);
  webrequest.contenttype = "application/soap+xml; charset=utf-8";
   webrequest.method = "post";
   using (stream requeststream = webrequest.getrequeststream())
   {
   byte[] parambytes = encoding.utf8.getbytes(datastr.tostring());
   requeststream.write(parambytes, 0, parambytes.length);
   }
   //response
   webresponse webresponse = webrequest.getresponse();
   using (streamreader mystreamreader = new streamreader(webresponse.getresponsestream(), encoding.utf8))
   {
   string result = "";
   return result = mystreamreader.readtoend();
  }
 
  }
  catch (exception ex)
  {
   throw ex;
  }
 
  }

但是上述方式还是存在问题,如果webservice需要校验header里面的用户名与密码时,虽然信封中封装了用户名和密码,但是一样调用不成功,同样的信封内容,使用soap ui却能成功。可见代码调用和soap ui调用还是有一定区别的。

另外一种方式是使用动态编译:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
uri uri = new uri(txt_url.text);
  webrequest webrequest = webrequest.create(uri);
  webrequest.credentials = new networkcredential("hjq", "123");
  system.io.stream requeststream = webrequest.getresponse().getresponsestream();
  // get a wsdl file describing a service
  servicedescription sd = servicedescription.read(requeststream);
  string sdname = sd.services[0].name;
  // add in tree view
 
 
  // initialize a service description servimport
  servicedescriptionimporter servimport = new servicedescriptionimporter();
  servimport.addservicedescription(sd, string.empty, string.empty);
  servimport.protocolname = "soap";
 
  servimport.codegenerationoptions = codegenerationoptions.generateproperties;
 
 
  codenamespace namespace = new codenamespace();
  codecompileunit codecompileunit = new codecompileunit();
  codecompileunit.namespaces.add(namespace);
  // set warnings
  servicedescriptionimportwarnings warnings = servimport.import(namespace, codecompileunit);
 
  if (warnings == 0)
  {
   stringwriter stringwriter = new stringwriter(system.globalization.cultureinfo.currentculture);
   microsoft.csharp.csharpcodeprovider prov = new microsoft.csharp.csharpcodeprovider();
   prov.generatecodefromnamespace(namespace, stringwriter, new codegeneratoroptions());
 
 
 
   // compile the assembly with the appropriate references
   //string[] assemblyreferences = new string[2] { "system.web.services.dll", "system.xml.dll" };
   string[] assemblyreferences = new string[3] {"system.dll", "system.dll", "system.dll" };
   compilerparameters param = new compilerparameters(assemblyreferences);
   param.generateexecutable = false;
   param.generateinmemory = true;
   param.treatwarningsaserrors = false;
   param.warninglevel = 4;
 
   compilerresults results = new compilerresults(new tempfilecollection());
   results = prov.compileassemblyfromdom(param, codecompileunit);
   assembly assembly = results.compiledassembly;
   type service = assembly.gettype(sdname);
 
   methodinfo[] methodinfo = service.getmethods();
   foreach (methodinfo t in methodinfo)
   {
   if (t.name == "discover")
    break;
 
   if (t.name == cbmethods.text)
   {
    //messagebox.show(t.tostring());
    //invoke method
    
    object obj = activator.createinstance(service);
    object response = t.invoke(obj, new object[] { "1","2","3"});
    messagebox.show("result = " + response.tostring());
    break;
   }
   }
  }
 
  }
  catch (exception ex)
  {
  messagebox.show(ex.message);
  }

从这种方式可以看出,首先是访问wsdl的地址取回webservice的结构,然后采取动态编译生成本地程序集的方式去调用webservice,和web引用自动生成reference.cs的原理类似;但是明显这种方式也存在问题,如果webservice的header中需要校验用户名与密码怎么处理呢?

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对服务器之家的支持。

原文链接:http://www.cnblogs.com/goldenbridge/p/7781482.html