Spring boot 整合CXF开发web service

汪阿苏
2023-12-01

前言

说起web service最近几年restful大行其道,大有取代传统soap web service的趋势,但是一些特有或相对老旧的系统依然使用了传统的soap web service,例如银行、航空公司的机票查询接口等。

目前就遇到了这种情况,需要在系统中查询第三方提供的soap web service接口,也就是说要将它整合进现有的系统当中。

Spring整合CXF本来十分简单,但是因为使用了Spring boot,不想用以前xml一堆配置的方式,那么能否按照Spring boot的风格优雅的进行整合呢?

答案当然是肯定的,但是遍查网上几乎没有这方面的资料,折腾过后觉得还是有必要记录一下,虽然它显得非常的简单。

添加依赖

Maven的项目,首先当然是添加依赖了,除了原先Spring boot的依赖之外,还需要添加cxf依赖:

  1. <dependency>
  2. <groupId>org.apache.cxf</groupId>
  3. <artifactId>cxf-rt-frontend-jaxws</artifactId>
  4. <version>3.1.6</version>
  5. </dependency>
  6. <dependency>
  7. <groupId>org.apache.cxf</groupId>
  8. <artifactId>cxf-rt-transports-http</artifactId>
  9. <version>3.1.6</version>
  10. </dependency>

编写业务代码

这里我们以查询用户信息为例,创建一个自定义的User对象:

  1. public class User implements Serializable {
  2. private static final long serialVersionUID = -5939599230753662529L;
  3. private Long userId;
  4. private String username;
  5. private String email;
  6. private Date gmtCreate;
  7. //getter setter ......
  8. }

接下来创建一个用于提供web service服务的用户接口,共两个方法getNamegetUser,一个返回普通的String,一个返回自定义对象:

  1. @WebService
  2. public interface UserService {
  3. @WebMethod
  4. String getName(@WebParam(name = "userId") Long userId);
  5. @WebMethod
  6. User getUser(Long userId);
  7. }

有接口当然要有业务代码实现了,这里我们只做简单的演示:

  1. public class UserServiceImpl implements UserService {
  2. private Map<Long, User> userMap = new HashMap<Long, User>();
  3. public UserServiceImpl() {
  4. User user = new User();
  5. user.setUserId(10001L);
  6. user.setUsername("liyd1");
  7. user.setEmail("liyd1@qq.com");
  8. user.setGmtCreate(new Date());
  9. userMap.put(user.getUserId(), user);
  10. user = new User();
  11. user.setUserId(10002L);
  12. user.setUsername("liyd2");
  13. user.setEmail("liyd2@qq.com");
  14. user.setGmtCreate(new Date());
  15. userMap.put(user.getUserId(), user);
  16. user = new User();
  17. user.setUserId(10003L);
  18. user.setUsername("liyd3");
  19. user.setEmail("liyd3@qq.com");
  20. user.setGmtCreate(new Date());
  21. userMap.put(user.getUserId(), user);
  22. }
  23. @Override
  24. public String getName(Long userId) {
  25. return "liyd-" + userId;
  26. }
  27. @Override
  28. public User getUser(Long userId) {
  29. return userMap.get(userId);
  30. }
  31. }

发布服务

接口和业务代码我们都写完了,剩下的就是发布服务了,也就是Spring boot和cxf的整合。

其实这二者的整合十分的简单,比以前xml的方式更加的简洁,所有相关的代码如下:

  1. @Configuration
  2. public class CxfConfig {
  3. @Bean
  4. public ServletRegistrationBean dispatcherServlet() {
  5. return new ServletRegistrationBean(new CXFServlet(), "/soap/*");
  6. }
  7. @Bean(name = Bus.DEFAULT_BUS_ID)
  8. public SpringBus springBus() {
  9. return new SpringBus();
  10. }
  11. @Bean
  12. public UserService userService() {
  13. return new UserServiceImpl();
  14. }
  15. @Bean
  16. public Endpoint endpoint() {
  17. EndpointImpl endpoint = new EndpointImpl(springBus(), userService());
  18. endpoint.publish("/user");
  19. return endpoint;
  20. }
  21. }

可以看到从配置cxf到发布服务,都只需要一二行代码,出乎意料的简单吧?

到这里我们所有的操作就都完成了,启动Spring boot,访问 http://localhost:8080/soap/user?wsdl

可以看到有相关的wsdl描述信息输出了,说明服务已经发布了。

调用服务

发布了web service服务,那怎么调用呢,像整合一些第三方接口也是先有调用才后有发布啊?

调用soap web service,一般的方法是根据wsdl生成客户端代码,整合之后就可以像调用本地接口一样使用了。

但是我个人不怎么喜欢这种方式,每个接口都要生成一次还有一堆的代码,感觉比较麻烦。

相对更喜欢传入方法名调用的方式,显得清爽而简洁,以下就是所有代码了:

  1. JaxWsDynamicClientFactory dcf = JaxWsDynamicClientFactory.newInstance();
  2. Client client = dcf.createClient("http://localhost:8080/soap/user?wsdl");
  3. Object[] objects = client.invoke("getUser", 10002L);
  4. //输出调用结果
  5. System.out.println(objects[0].getClass());
  6. System.out.println(objects[0].toString());

这种方式要注意的就是,如果调用的服务接口返回的是一个自定义对象,那么结果Object[]中的数据类型就成了这个自定义对象(组件帮你自动生成了这个对象),

但是你本地可能并没有这个类,所以需要自行转换处理,最简单的是新建一个跟返回结果一模一样的类进行强转,当然更好的方式是封装一个通用的,这个不是本文主题就不在这深入讨论了。

 类似资料: