使用xfire快速发布WebService接口

作者:赵磊

博客:http://elf8848.iteye.com

 

一、WebService应用场景简述

  WebService的主要目标是跨平台的可互操作性。为了达到这一目标,WebService完全基于XML(可扩展标记语言)、XSD(XMLSchema)等独立于平台、独立于软件供应商的标准,是创建可互操作的、分布式应用程序的新平台。
  1、跨防火墙的通信,基本HTTP协议,走80端口,多公司,多机房通信时不用为防火墙烦恼
  2、应用程序集成,把用不同语言写成的、在不同平台上运行的各种程序集成起来,也就是异构系统的系统。比如服务端是java的webService实现XFire做的服务端,客户端是.net的
 webService实现或C++的 webService实现都可以。


   不适合使用WebService的场景
  1、单机应用程序
  2、局域网的同构应用程序
  总之,只要从应用程序结构的角度看,有别的方法比WebService更有效、更可行,那就不要用WebService,WebService的性能是多种通信协议中最慢的.

 

二、XFire简介

   XFire是新一代的Java WebService引擎,XFire使得在JavaEE应用中发布Web服务变得轻而易举。和其他Web服务引擎相比,XFire的配置非常简单,可以非常容易地和Spring集成,它使得Java开发人员终于可以获得和.Net开发人员一样的开发效率。

  

   XFire是codeHaus组织提供的一个开源框架,它构建了POJO和SOA之间的桥梁,主要特性就是支持将POJO通过非常简单的方式发布成Web服务,这种处理方式不仅充分发挥了POJO的作用,简化了Java应用转化为Web服务的步骤和过程,也直接降低了SOA的实现难度,为企业转向SOA架构提供了一种简单可行的方式。

  

   XFire 支持将Web服务绑定到POJO、XMLBeans、JAXB1.1、JAXB2.0和Castor;

  支持基于HTTP、JMS、XMPP等多种协议访问Web服务;

  支持多种Web服务业界重要标准如SOAP、WSDL、Web服务寻址(WS-Addressing)、Web服务安全(WS-Security)等;

  支持JSR181,可以通过JDK5配置Web服务;

  高性能的SOAP实现;

  服务器端、客户端代码辅助生成;

  对Spring、Pico、Plexus等项目的支持等。

 

   XFire是一种基于Servlet技术的SOA应用开发框架,需要Servlet容器的支持。XFire支持在多种Servlet容器中运行,包括Websphere、Weblogic、TOMCAT等。支持JDK 1.4以上(含1.4).


   XFire框架目前的最新版本是1.2.6,可以访问http://xfire.codehaus.org 下载XFire框架的安装包,下载时请选择“全部二进制发布包”,而不仅仅是“XFirejar文件".下载地址:http://xfire.codehaus.org/Download

 

 

 

三、快速发布WebService接口:

 

1.下载XFire

2.用Eclipse创建一个Web项目,

3.导入xfire-distribution-1.2.6\xfire-1.2.6\lib中的所以jar包

4.导入xfire-distribution-1.2.6\xfire-1.2.6\xfire-all-1.2.6.jar包

5.创建实体类User类,这个类将被通过SOAP协议传输.

Java代码   收藏代码
  1. package webService;  
  2. import java.util.Date;  
  3. public class User {  
  4.     private static final long serialVersionUID = 6517808321041980976L;  
  5.     private Long userId;  
  6.     private String accountId;  
  7.     private String userName;  
  8.     private Date lastLogin;  
  9.   
  10.     public String getAccountId() {  
  11.         return accountId;  
  12.     }  
  13.   
  14.     public void setAccountId(String accountId) {  
  15.         this.accountId = accountId;  
  16.     }  
  17.   
  18.     public Date getLastLogin() {  
  19.         return lastLogin;  
  20.     }  
  21.   
  22.     public void setLastLogin(Date lastLogin) {  
  23.         this.lastLogin = lastLogin;  
  24.     }  
  25.   
  26.     public Long getUserId() {  
  27.         return userId;  
  28.     }  
  29.   
  30.     public void setUserId(Long userId) {  
  31.         this.userId = userId;  
  32.     }  
  33.   
  34.     public String getUserName() {  
  35.         return userName;  
  36.     }  
  37.   
  38.     public void setUserName(String userName) {  
  39.         this.userName = userName;  
  40.     }  
  41. }  
 

6.编写接口UserService,这个接口暴露两个方法

Java代码   收藏代码
  1. package webService.server;  
  2. import webService.User;  
  3.   
  4. public interface UserService {  
  5.     public User queryUserByAccoutId(String accountId);  
  6.   
  7.     public void createUser(User user);  
  8. }  
 

 

7.实现接口UserServiceImpl, 实现接口UserService.

Java代码   收藏代码
  1. package webService.server;  
  2. import java.rmi.RemoteException;  
  3. import java.util.Date;  
  4. import org.apache.commons.logging.Log;  
  5. import org.apache.commons.logging.LogFactory;  
  6. import webService.User;  
  7.   
  8. public class UserServiceImpl implements UserService {  
  9.   
  10.     private static final Log log = LogFactory.getLog(UserServiceImpl.class);  
  11.   
  12.     public void createUser(User user) {  
  13.         log.debug("createUser user=" + user);  
  14.     }  
  15.   
  16.     public User queryUserByAccoutId(String key) {  
  17.         log.info("key=" + key);  
  18.         User user = new User();  
  19.         user.setAccountId("testAccount");  
  20.         user.setLastLogin(new Date());  
  21.         user.setUserId(new Long(123L));  
  22.         user.setUserName("用户名");  
  23.         return user;  
  24.     }  
  25. }  
 

8.修改web.xml文件,目的是注册servlet, 发布暴露WSDL.

Xml代码   收藏代码
  1. <?xml version="1.0" encoding="UTF-8"?>  
  2. <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"   
  3. xmlns="http://java.sun.com/xml/ns/javaee"   
  4. xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"   
  5. xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"   
  6. id="WebApp_ID" version="2.5">  
  7.   <context-param>  
  8.     <param-name>contextConfigLocation</param-name>  
  9.     <param-value>classpath:applicationContext.xml</param-value>  这是spring的配置文件  
  10.   </context-param>  
  11.   <listener>  
  12.     <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>  
  13.   </listener>  
  14.   <servlet>  
  15.     <servlet-name>xfire</servlet-name>  
  16.     <servlet-class>org.codehaus.xfire.spring.XFireSpringServlet</servlet-class>  
  17.   </servlet>  
  18.   <servlet-mapping>  
  19.     <servlet-name>xfire</servlet-name>  
  20.     <url-pattern>/service/*</url-pattern>  注意这个地址,一会要访问这个地址,查看wsdl  
  21.   </servlet-mapping>  
  22. </web-app>  
 

9.修改spring配置文件applicationContext.xml

Xml代码   收藏代码
  1. <?xml version="1.0" encoding="UTF-8"?>   
  2. <!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN" "http://www.springframework.org/dtd/spring-beans.dtd">  
  3. <beans>  
  4.     <import resource="classpath:org/codehaus/xfire/spring/xfire.xml" />  
  5.     <bean id="baseWebService" class="org.codehaus.xfire.spring.remoting.XFireExporter"  
  6.         lazy-init="false" abstract="true">  
  7.         <property name="serviceFactory" ref="xfire.serviceFactory" />  
  8.         <property name="xfire" ref="xfire" />  
  9.     </bean>  
  10.     <bean id="userWS" class="webService.server.UserServiceImpl"></bean>  这是接口实现类  
  11.     <bean id="userService" parent="baseWebService">    
  12.         <property name="serviceBean" ref="userWS" />  
  13.         <property name="serviceClass" value="webService.server.UserService" />  这是接口  
  14.     </bean>  
  15. </beans>  
 

 

10. 启动TOMCAT, 

访问地址:   http://localhost:8080/应用根/service   查看wsdl

如果能查看到wsdl,服务端就OK了

也可以使用Eclipse中的WTP中的Web Service Explore测试SOAP方式,具体操作方法,看最后面下载中的PDF文件.


下面生成客户端

11.通过Ant自动生成客户端, 在web目录 (你可能是WebRoot)创建build.xml文件,并运行这个文件生成代码,注意根据你的情况修改哦.

Xml代码   收藏代码
  1. <?xml version="1.0"?>  
  2. <project name="wsgen" default="wsgen" basedir=".">  
  3.     <path id="classpathId">  
  4.         <fileset dir="./WEB-INF/lib">  
  5.             <include name="*.jar" />  
  6.         </fileset>  
  7.     </path>  
  8.     <taskdef classpathref="classpathId" name="wsgen" classname="org.codehaus.xfire.gen.WsGenTask">  
  9.     </taskdef>  
  10.     <target name="wsgen" description="generate client">  
  11.         <wsgen outputDirectory="../src/" wsdl="http://127.0.0.1:8080/RR_Test/service/UserService?wsdl" binding="xmlbeans" package="webService.client" overwrite="true" />  
  12.                         请自行修改这几个参数,outputDirectory是src目录位置,    wsdl是wsdl的url改成你自己的地址,    package是生成的代码存放的包名  
  13.  </target>  
  14. </project>  
 

 

12. 编写测试类WebServiceClientTest

Java代码   收藏代码
  1. package webService.client;  
  2.   
  3. import java.net.MalformedURLException;  
  4. import java.util.concurrent.ExecutorService;  
  5. import java.util.concurrent.Executors;  
  6. import org.apache.log4j.Logger;  
  7. import org.apache.xmlbeans.XmlObject;  
  8. import org.codehaus.xfire.client.Client;  
  9. import org.codehaus.xfire.client.XFireProxyFactory;  
  10. import org.codehaus.xfire.service.binding.ObjectServiceFactory;  
  11. import org.codehaus.xfire.transport.http.HttpTransport;  
  12. import webService.User;  
  13. import webService.server.UserService;  
  14. import org.codehaus.xfire.service.Service;  
  15.   
  16. /** 
  17.  * 说明: WebService客户端, 是测试工具类 
  18.  *  
  19.  * 如何查看测试所用时间? 
  20.  * 请看log4j.properties文件,出输出日志到d:\\mina.log, 
  21.  * log4j记录每一次请求响应的日志,包括时间. 
  22.  * 请使用结束的时间减去开始的时间,得到总用时.   
  23.  */  
  24. public class WebServiceClientTest {  
  25.     private static Logger logger = Logger.getLogger(WebServiceClientTest.class);  
  26.   
  27.     // WebService服务端地址,依据实际情况修改  
  28.     private static String baseUrl = "http://127.0.0.1:8080/RR_Test";  
  29.       
  30.     //服务器返回的数据量大小,请根据需要修改 ,有三个值可选 4k,15k,50k  
  31.     private static String key="15k";  
  32.       
  33.     // 这里修改并发数  
  34.     private static int threadSize=100;  
  35.     // 这里修改重复次数  
  36.     private static int count=100;  
  37.   
  38.     /** 
  39.      * 主测试方法 
  40.      *  
  41.      * @throws MalformedURLException 
  42.      */  
  43.     public static void main(String[] args) {  
  44.           
  45.         test1();  
  46.           
  47. //      ExecutorService exec = Executors.newFixedThreadPool(500);  
  48. //      logger.info("WebService client说:开始测试");  
  49. //      for (int i = 0; i < threadSize; i++) {  
  50. //          exec.execute(new Runnable() {  
  51. //              public void run() {  
  52. //                  for (int j = 0; j < count; j++) {  
  53. //                      test2();  
  54. //                  }  
  55. //              }  
  56. //          });  
  57. //      }  
  58. //      exec.shutdown();  
  59.     }  
  60.   
  61.     /** 
  62.      * 测试1 
  63.      */  
  64.     public static void test1() {  
  65.   
  66.         UserServiceClient client = new UserServiceClient();  
  67.         String url = baseUrl + "/service/UserService";  
  68.         XmlObject xmlObject = client.getUserServiceHttpPort(url).queryUserByAccoutId("CN20106195184");  
  69.         System.out.println(xmlObject);  
  70.     }  
  71.   
  72.     /** 
  73.      * 测试2 
  74.      */  
  75.     public static void test2() {  
  76.         try {  
  77.             Service serviceModel = new ObjectServiceFactory().create(UserService.class);  
  78.             UserService service = (UserService) new XFireProxyFactory().create(serviceModel, baseUrl  
  79.                     + "/service/UserService");  
  80.             Client client = Client.getInstance(service);  
  81.             client.setProperty("mtom-enabled""true");  
  82.             client.setProperty(HttpTransport.CHUNKING_ENABLED, "true");  
  83.             User user = service.queryUserByAccoutId(key);  
  84.             logger.info("userId=" + user.getUserId() + ", userName长度=" + user.getUserName().length()  
  85.                     + ", lastLogin=" + user.getLastLogin());  
  86.         } catch (MalformedURLException e) {  
  87.             e.printStackTrace();  
  88.         }  
  89.     }  
  90. }  
 

13.注意修改测试类中的URL , 要与你的程序发布情况一致, 主要是 IP , 端口, 应用根

这里log4j的东西,省略了, 你自己加上吧.

运行main()方法, 看结果吧

 

 

 

 

四、常见异常解决方法:

ps:如果出现class path resource [META-INF/xfire/services.xml] cannot be opened because it does not exist
修改方法:把META-INF文件夹copy到工程的build的classes目录下。

出现异常org.springframework.beans.factory.BeanDefinitionStoreException: Unrecognized xbean namespace mapping:http://XFire.codehaus.org/config/1.0 
查阅官方文档,发现 xmlns="http://xfire.codehaus.org/config/1.0 "> 是全部小写,立即替换,解决!

你可能感兴趣的:(java,java,java,webservice,webservice,webservice,xfire)