前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >不同的.Net版本客户端软件调用Java Web Service区别

不同的.Net版本客户端软件调用Java Web Service区别

作者头像
程序你好
发布2018-08-09 15:27:14
1.4K0
发布2018-08-09 15:27:14
举报
文章被收录于专栏:程序你好程序你好

最近的系统中需要.Net开发的离线端软件通过Web Service技术和Java开发的在线系统进行数据交互。

在这个过程中碰到不少问题,写出来供大家参考。

Java这边在线系统基于easyUI,String Boot,mybatis技术,前后端完全分离的架构。

Java Web Service实现:

先来吐槽一下,其实如果使用Restful Json的方式与客户端交互,我们在线系统本身就支持,客户端也需要去处理soap版本的问题,实现起来更方便。可我说了不算。。。

1、Java端的代码很简单,引入包,加上webservice注解,写代码实现逻辑。

代码语言:javascript
复制
@WebService(targetNamespace = "http:/****/", endpointInterface = "******.TaskWebService")
@Component
public class TaskWebServiceImpl implements TaskWebService {
@Autowired
private SqlSessionTemplate session;
@Autowired
private QueryService queryService;
@Autowired
private EntityDao entityDao;
@Override
public List<TaskOrderInfo> getTaskOrders() {
TaskOrderReturn result = new TaskOrderReturn();
User user = User.getCurUser();
Org org = user.getOrg();
String orgLevel = org.getOrgLevel();
Map<String, Object> map = getMapByUser(user);
if (map.isEmpty()) {
return new ArrayList<TaskOrderInfo>();
}
               ......
         }

2、.Net 4.0客户端调用

在.Net Framework 2.0和4.0里生成的proxy代理类是不同的。

先来看4.0:

public partial class TaskWebServiceClient : System.ServiceModel.ClientBase

代理类继承自WCF服务Client对象,这个类是在.Net 3.0中引入WCF框架的时候添加进来的。

System.ServiceModel.ClientBase:提供用于创建可以调用服务的Windows通信基础(WCF)客户端对象的基本实现。

主要用于调用WCF创建的各种Binding类型的服务,当然也包括BasicHttpBinding也就是Soap 1.1。

接下来使用此代理类调用web服务。

这里碰到一个问题,Soap版本的问题,因为自动生成的代理类只支持soap1.1,可我们Java web服务是基于高版本的soap协议, 结果调研失败。

Java服务端改回1.1版。

接着继续。

又发现一个坑,本来应该从.Net Client 传送到Web Service上的参数,服务器接收不到。

使用Soap工具跟踪,发现soap消息里根本就没有。

后来发现生成的代理类中,所有的值类型字段都增加了一个变量名+ Specified命名的布尔字段。

默认这些字段是False,这种情况下,在调用web服务时,这些字段的值不会加入到soap消息里。

修改代码,设置这些字段为True。

代码语言:javascript
复制
                     para.startTime = DateTime.Parse("2018-05-01");
                    para.startTimeSpecified = true;
                    para.endTime = DateTime.Parse("2018 -07-01");
                    para.endTimeSpecified = true;

服务器端设置了在http header中加入了,access token进行访问控制。

.Net 4.0中在http header中加入token代码:

代码语言:javascript
复制
using (var wsClient = new wsClient.UpDownWebServiceClient())
              
            {
                using (OperationContextScope scope = new OperationContextScope(wsClient.InnerChannel))
                {
                    var httpRequestProperty = new HttpRequestMessageProperty();
                    httpRequestProperty.Headers["Authorization"] = accessTokenKey
                    OperationContext.Current.OutgoingMessageProperties[HttpRequestMessageProperty.Name] = httpRequestProperty;
                    downloadParam para = new downloadParam();
                    para.pageNum = 1;
                    para.pageNumSpecified = true;
                    para.pageSize = 10;
                    para.pageSizeSpecified = true;
                    para.startTime = DateTime.Parse("2018-05-01");
                    para.startTimeSpecified = true;
                    para.endTime = DateTime.Parse("2018 -07-01");
                    para.endTimeSpecified = true;

                    var list = wsClient.download(para);                 
                }

            }

调用成功!

可发现一个奇怪的问题,有个别的Web Service API行为很奇怪。这个API是返回一组数据列表,客户端调用正常,soap消息监控正常,服务器端执行正常,

可是返回到.net客户端的对象一直是空。而使用Java客户端调用和返回都是正常的。

经过监视soap消息,发现消息中其实已经返回了对象数据。怀疑是在soap消息反序列化成.net对象时候出了问题。可这个问题一直没找到原因。

3、使用.net2.0调用服务

在2.0中代理类从 System.Web.Services.Protocols.SoapHttpClientProtocol 类继承而来。

使用access token

重载代理类的 GetWebRequest方法,在http header中加入access token

代码语言:javascript
复制
 protected override System.Net.WebRequest GetWebRequest(Uri uri)
        {
            var request = base.GetWebRequest(uri);
            request.Headers.Add("Authorization", this.Authorization); // <----
            return request;
        }

如果需要在soap header加入内容:

从System.Web.Services.Protocols.SoapHeader类继承,定义自己的soapheader类:MySoapHeader

在代理类中增加属性:

代码语言:javascript
复制
 private MySoapHeader mySoapHeaderValueField;
        public MySoapHeader MySoapHeaderValue
        {
            get
            {
                return this.mySoapHeaderValueField;
            }
            set
            {
                this.mySoapHeaderValueField = value;
            }
        }

在 web方法上增加attribute声明:

代码语言:javascript
复制
/// <remarks/>
        [System.Web.Services.Protocols.SoapHeaderAttribute("MySoapHeaderValue")]
        [System.Web.Services.Protocols.SoapDocumentMethodAttribute("", RequestNamespace="http://****/", ResponseNamespace="http://****/", Use=System.Web.Services.Description.SoapBindingUse.Literal, ParameterStyle=System.Web.Services.Protocols.SoapParameterStyle.Wrapped)]
        [return: System.Xml.Serialization.XmlElementAttribute("return", Form=System.Xml.Schema.XmlSchemaForm.Unqualified)]
        public downloadReturn download([System.Xml.Serialization.XmlElementAttribute(Form=System.Xml.Schema.XmlSchemaForm.Unqualified)] downloadParam arg0) {
            object[] results = this.Invoke("download", new object[] {
                        arg0});
            return ((downloadReturn)(results[0]));
        }

调用成功,而且使用2.0调用web 服务并没有发现4.0版本发现的问题。

总结

说实话,本来应该是个很简单的问题,结果碰到一堆坑。建议大家使用Restful /Json方式进行服务器和客户端数据交互,尤其是在跨语言的情况行下,更方便,更简单,也不需要关心soap版本之类的东东。

请关注公众号:程序你好

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2018-08-01,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 程序你好 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 1、Java端的代码很简单,引入包,加上webservice注解,写代码实现逻辑。
  • 2、.Net 4.0客户端调用
  • 3、使用.net2.0调用服务
  • 总结
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档