我算是个AXIS2的初学者,以前用它来调用.net环境下的ws,几乎每遇到困难。用工具一路生成(XMLBeans绑定),然后就在程序中直接调用,因此也没有对其作深入的了解。
最近项目需要调用WCF架构下发布的ws,很不顺利,走了很多弯路,现在稍微有了点头绪......
使用最新版本的axis2携带的工具根据wsdl生成java代码,生成步骤网上有很多介绍,很简单的,我用的是Intellij IDEA的插件,稍微配置下搞定了。
注意:axis2版本一定要1.2以上的,否则会出错,之前版本并不能很好的兼容soap1.2规范。尝试过XFire最新版本1.2.6,不过很可惜,生成不了代码,而其动态调用的方式对含有复杂数据类型的ws来说太繁琐了。
比较而言,AXIS2对ms产品的支持力度还是值得肯定的。
然后在测试程序中加入客户端代码调用。调用的代码就不写了,和原先的调用方法没什么两样。
运行时出现如下异常:
java 代码
- org.apache.axis2.AxisFault: Transport out has not been set
- at org.apache.axis2.engine.AxisEngine.send(AxisEngine.java:440)
- at org.apache.axis2.description.OutInAxisOperationClient.send(OutInAxisOperation.java:329)
- at org.apache.axis2.description.OutInAxisOperationClient.execute(OutInAxisOperation.java:294)
- at movie.MovieManagerStub.GetMovieByID(MovieManagerStub.java:7294)
- at movie.TestMovieClient.main(TestMovieClient.java:22)
- at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
- at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
- at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
- at java.lang.reflect.Method.invoke(Method.java:585)
- at com.intellij.rt.execution.application.AppMain.main(AppMain.java:90)
网上对这个异常描述很少,无奈之下,下载了axis2源代码手动跟踪......
终于发现问题:
在wsdl描述文件中有如下的xml语句
xml 代码
- <wsdl:service name="MovieManager">
- <wsdl:port name="MetadataExchangeTcpBinding_MovieManager" binding="i0:MetadataExchangeTcpBinding_MovieManager">
- <soap12:address location="net.tcp://10.0.0.249:8002/BusinessPlatform/MovieManager" />
- <wsa10:EndpointReference>
- <wsa10:Address>net.tcp://10.0.0.249:8000/BusinessPlatform/MovieManagerwsa10:Address>
- wsa10:EndpointReference>
- wsdl:port>
- wsdl:service>
就是"net.tcp://"这段,axis2对net.tcp这个传输方式不能辨识。
axis2的kernal包中的org\apache\axis2\deployment\axis2_default.xml文件对支持的传输方式作了配置:
xml 代码
- <transportReceiver name="http"
- class="org.apache.axis2.transport.http.SimpleHTTPServer">
- <parameter name="port" locked="false">6060parameter>
-
-
-
- transportReceiver>
-
- <transportReceiver name="tcp"
- class="org.apache.axis2.transport.tcp.TCPServer">
- <parameter name="port" locked="false">6061parameter>
-
-
-
- transportReceiver>
-
-
-
-
-
- <transportSender name="jms"
- class="org.apache.axis2.transport.jms.JMSSender"/>
- <transportSender name="tcp"
- class="org.apache.axis2.transport.tcp.TCPTransportSender"/>
- <transportSender name="local"
- class="org.apache.axis2.transport.local.LocalTransportSender"/>
- <transportSender name="http"
- class="org.apache.axis2.transport.http.CommonsHTTPTransportSender">
- <parameter name="PROTOCOL" locked="false">HTTP/1.1parameter>
- <parameter name="Transfer-Encoding" locked="false">chunkedparameter>
- transportSender>
- <transportSender name="https"
- class="org.apache.axis2.transport.http.CommonsHTTPTransportSender">
- <parameter name="PROTOCOL" locked="false">HTTP/1.1parameter>
- <parameter name="Transfer-Encoding" locked="false">chunkedparameter>
- transportSender>
核心 程序会根据这个配置来获得TransportSender这样的一个实例用于数据传输。
如果想要支持"net.tcp",可能需要扩展这个配置文件,甚至于需要自己去编写sender类。
对于wcf不是很了解,不知道"net.tcp"是否很换成其他协议,我需要和对方技术人员沟通下,看有无其他的解决方法。
在对方调整后,wsdl描述中不再出现"net.tcp"之类ms特有的通讯协议,生成和调用顺利通过。