java web服务
如果在最初的项目体系结构中未考虑Web服务,这将不会有些棘手。 您可能会涉及不同种类的项目体系结构并使用不同种类的技术。 作为开发人员,由于可能存在很多依赖关系,因此您不能更改某些体系结构和配置。
Internet上的大多数教程都介绍了创建Web服务的基本内容。 有时,“ Hello world”应用程序或有时可能是像这样的简单计算器。 这些教程非常有助于您对Web服务有基本的了解。 但是现实世界的场景要比这复杂十倍,并且在遵循此类教程时必须面对困难。
实际情况
在本教程中,我将解释“我们如何真正解决您的主管提出的现实要求”。 我将解释我最近遇到的那种情况。
一个医疗保健组织在岛上经营着许多药房。 他们有一个处理所有库存,定价和开票,发行药房物品等的Web应用程序。他们需要通过Web服务公开药房项目的价格,以便药房中的客户应用程序可以通过Web服务访问这些产品。
他们的Web应用程序是在struts2,Spring和Hibernate的集成环境中开发的。 它具有所有Spring管理的DAO类以及服务类。 该应用程序使用spring的自动接线技术,组件扫描,事务管理等。在这种背景下,我需要将药房价格作为Web服务公开。 那就是我们当前药房服务中的一些方法需要通过网络服务公开给外部。
我将向您展示,如何通过对现有项目进行最小的修改来实现这种要求。
附加图书馆
我将使用JAX-WS实现Web服务。 我在项目中使用了JAX-WS 2.2。 您可以从此处下载所需的JAX-WS版本。 这提供了一些可用于生成Web服务及其客户端内容的工具。 下载所需的库版本后,将其解压缩到本地计算机中的某些位置。 我已将其放在主文件夹中。
即:/home/semika/jaxws-ri-2.2
实施网络服务
我已经有了Spring托管的服务类和DAO类(用于药房项目),该类已在Web应用程序内部使用。 那些没有暴露在外面。 假设,我们需要公开findAll()方法,该方法以Web服务的形式返回“ PharmacyService”接口的“ PharmacyItem”列表。
为方便起见,我将向您展示“ PharmacyServiceImpl” java类,该类用于执行常规药房项目操作。 这是通常的Springman豆。 请记住,此类的方法目前只能用于我们的Web应用程序的内部操作。 这些不作为Web服务公开。
PharmacyServiceImpl.java
/*** */
package com.slauto.service.pharmacy.impl;import java.util.List;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;import com.slauto.exceptions.ServiceException;
import com.slauto.model.pharmacy.PharmacyItem;
import com.slauto.persist.pharmacy.api.PharmacyItemDao;
import com.slauto.service.pharmacy.api.PharmacyItemService;/*** @author semika**/
@Service('pharmacyItemService')
public class PharmacyItemServiceImpl implements PharmacyItemService {@Autowiredprivate PharmacyItemDao pharmacyItemDao;@Overridepublic List<Pharmacyitem> findAll() throws ServiceException {return pharmacyItemDao.findAllPharmacyItems();}
}
如您所见,我在实现类中具有“ PharmacyItemDao”的自动连线实例。 众所周知,这是一个Spring托管服务实现类。
接下来,我们将为上面的spring托管服务bean实现Web服务端点类,以将其方法公开为Web服务方法。 为了清楚起见,我创建了一个单独的类作为Web服务端点。
PharmacyItemServiceEndPoint.java
package com.slauto.service.ws;
import javax.jws.WebMethod;
import javax.jws.WebService;import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;import com.slauto.exceptions.DataAccessException;
import com.slauto.model.pharmacy.PharmacyItem;
import com.slauto.service.pharmacy.api.PharmacyItemService;/*** @author semika**/
@WebService(serviceName='pharmacyItemService')
public class PharmacyItemServiceEndPoint {@WebMethodpublic List<Pharmacyitem> findAll() throws DataAccessException {ApplicationContext context = new ClassPathXmlApplicationContext('applicationContext.xml'); PharmacyItemService pharmacyItemService = (PharmacyItemService)context.getBean('pharmacyItemService');return pharmacyItemService.findAll();}
}
如您在这里看到的,我正在使用'pharmacyItemService'bean,我们的Web应用程序使用它来访问相关的服务方法。 “ pharmacyItemService”是一种常用的Spring mange bean,用于执行日常药房项目操作。 没什么特别的。 特别注意,指示此类的@WebService和@WebMethod注释用作Web服务端点。
在这里,我通过应用程序上下文获取服务类实例。 此外,我们可以通过扩展spring提供的' SpringBeanAutowiringSupport '的端点类,来使用spring自动连接Web服务端点类。 在这种情况下,我们不需要像上面所做的那样创建应用程序上下文实例。 我无法完成这项工作,这就是为什么我使用了上述技术。
使用“ SpringBeanAutowiringSupport ”,在部署Web服务时(说明如下),我遇到了一个异常,很难解决。 所以我选择了这项技术。 无论如何,我不喜欢上面使用的内容:)。
生成网络服务
我正在使用JAX-WS提供的apt , wsgen和wsimport工具来生成JAX-WS服务中使用的可移植工件。 “ build.xml ”文件的相关ant目标如下。
您可能需要在“ build.xml”文件顶部声明的以下属性。
属性:
<property name='tomcat.home' value='/home/semika/apache-tomcat-7.0.25' />
<property name='jaxws.home' value='/home/semika/jaxws-ri-2.2' />
<property name='build.classes.home' value='${basedir}/WEB-INF/classes' />
<property name='java.home' value='/home/semika/java/jdk1.6.0_30'>
类路径:
<path id='project.class.path'><pathelement location='${java.home}/../lib/tools.jar' /><fileset dir='${jaxws.home}/lib'><include name='*.jar' /></fileset><pathelement location='${basedir}/WEB-INF/classes' /><fileset dir='${basedir}/WEB-INF/lib' includes='*.jar' />
</path>
JAX-WS apt工具目标:
<target name='apt' depends='javac'><taskdef name='apt' classname='com.sun.tools.ws.ant.Apt'><classpath refid='project.class.path' /></taskdef><apt fork='true'debug='true'verbose='true'destdir='${basedir}/WEB-INF/classes'sourcedestdir='${basedir}/WEB-INF/src'sourcepath='${basedir}/WEB-INF/src'><classpath><path refid='project.class.path' /></classpath><option key='r' value='${basedir}/WEB-INF' /><source dir='${basedir}/WEB-INF/src'><include name='**/*.java'/></source></apt>
</target>
如果您想进一步了解JAX-WS提供的apt工具,请查看this 。 运行apt目标时,它将扫描源路径(src文件夹),并为使用@WebService注释的类生成所需的* .class和* .java文件。 在这种情况下,适用于“ PharmacyItemServiceEndPoint.java”。 如果查看“ PharmacyItemServiceEndPoint”所在的包,则可以看到它创建了一个名为“ jaxws ”的新包。 在该程序包中,我可以看到以下三个Java文件。
DataAccessExceptionBean.java
FindAll.java FindAllResponse.java
这些类由该工具生成,并且根据您的服务实现和Web服务涉及的依赖类而有所不同。 实际上,您不必担心这些生成的东西。
同样,您可以在/ WEB-INF / classes文件夹下看到相关的* .class文件。
JAX-WS wsgen工具目标:
<target name='wsgen' depends='apt'> <taskdef name='wsgen' classname='com.sun.tools.ws.ant.WsGen'><classpath path='project.class.path'/></taskdef><wsgen xendorsed='true'sei='com.slauto.service.ws.PharmacyItemServiceEndPoint'destdir='${basedir}/WEB-INF/classes' resourcedestdir='${wsdl.dir}'sourcedestdir='${basedir}/WEB-INF/src' keep='true'verbose='true'genwsdl='true'><classpath refid='project.class.path'/></wsgen>
</target>
wsgen工具将为我们的端点Web服务类生成WSDL文件。 运行此目标后,请查看$ {wsdl.dir}的位置。 您可以看到我们的WSDL文件已经生成。 如果您想进一步了解JAX-WS提供的wsgen工具,请研究this 。
部署Web服务
我想用通常的服务器启动来部署Web服务。 因此,我必须将以下配置添加到web.xml文件中。
<listener><listener-class>com.sun.xml.ws.transport.http.servlet.WSServletContextListener</listener-class>
</listener>
<servlet><servlet-name>pharmacyItemService</servlet-name><servlet-class>com.sun.xml.ws.transport.http.servlet.WSServlet</servlet-class><load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping><servlet-name>pharmacyItemService</servlet-name><url-pattern>/pharmacyItemService</url-pattern>
</servlet-mapping>
正如我之前告诉您的那样,当部署具有从' SpringBeanAutowiringSupport '扩展的端点类的Web服务时,它给出了一个例外。 因此,我决定通过应用程序上下文获取服务bean。 如果您能解决这种情况,请直接发布。
而且,您需要在WEB-INF文件夹下创建sun-jaxws.xml并声明Web服务端点,如下所示。
<?xml version='1.0' encoding='UTF-8'?>
<endpoints xmlns='http://java.sun.com/xml/ns/jax-ws/ri/runtime' version='2.0'> <endpointname='pharmacyItemService'implementation='com.slauto.service.ws.PharmacyItemServiceEndPoint'url-pattern='/pharmacyItemService'/>
</endpoints>
我正在使用apache tomcat 7.0.25部署Web服务。 您需要告诉tomcat启动tomcat时可以在哪里找到JAX-WS库。 您可以编辑位于CATALINA_HOME / conf文件夹中的“ catalina.properties ”文件。 查找common.loader属性。 大多数情况如下。
common.loader=${catalina.base}/lib,${catalina.base}/lib/*.jar,${catalina.home}/lib,${catalina.home}/lib/*.jar
对其进行如下修改。
common.loader=${catalina.base}/lib,${catalina.base}/lib/*.jar,${catalina.home}/lib,${catalina.home}/lib/*.jar,/home/semika/jaxws-ri-2.2/lib/*.jar
如您所见,我已经在逗号分隔列表的末尾添加了我的JAX-WS库路径。 这些类对tomcat内部类和tomcat容器中部署的所有Web应用程序均可见。 现在,您可以将.war文件复制到tomcat的webapps文件夹中,然后启动tomcat。 您的Web服务应已部署。 为了确认您的Web服务已正确部署,您可以检查它的WSDL 。 对于WSDL文件,您应该以以下格式检查URL。
http:// localhost:8080 / <您的上下文名称> / pharmacyItemService?wsdl
如果您可以在浏览器中看到WSDL文件,则说明您已完成,您已经成功完全部署了Web服务。
生成Web服务客户端
现在,我们已经部署了Web服务。 接下来,我们将看到如何编写客户端类以使用Java程序访问Web服务。 您可以创建一个简单的Java项目,并在其中使用以下build.xml文件。 我已经放置了完整的build.xml文件用于客户端生成。 我正在使用JAX-WS随附的wsimport工具来生成Web服务客户端工件。
<?xml version='1.0' encoding='utf-8' ?>
<project name='WS-client' default='wsimport' basedir='.'><property name='jaxws.home' value='/home/semika/jaxws-ri-2.2' /><property name='java.home' value='/home/semika/java/jdk1.6.0_30'/><path id='jaxws.classpath'><pathelement location='${java.home}/../lib/tools.jar' /><fileset dir='${jaxws.home}/lib'><include name='*.jar' /></fileset></path><target name='wsimport'> <taskdef name='wsimport' classname='com.sun.tools.ws.ant.WsImport'><classpath path='jaxws.classpath'/></taskdef><wsimport xendorsed='true'debug='true'verbose='true'keep='true'destdir='src'package='com.slauto.service'wsdl='http://localhost:8080/slautomanage/pharmacyItemService?wsdl'></wsimport></target>
</project>
如果您想进一步了解wsimport工具,可以查看this 。 运行上述目标后,只需看看在wsimport属性下指定的包即可。 您将看到生成的Java文件集。 您可以将它们编译并捆绑到单个client.jar文件中,并且可以与需要此Web服务的任何Java项目一起使用。
我创建了非常简单的Java类,以通过Web服务获取药房信息。
客户端程序
package com.slauto.client;import java.util.List;import com.slauto.service.PharmacyItem;
import com.slauto.service.PharmacyItemService;
import com.slauto.service.PharmacyItemServiceEndPoint;/*** @author semika**/
public class Client {/*** @param args*/public static void main(String[] args) {PharmacyItemService p = new PharmacyItemService();PharmacyItemServiceEndPoint ep = p.getPharmacyItemServiceEndPointPort();List<PharmacyItem> pharmacyItems = ep.findAll();for (PharmacyItem pharmacyItem : pharmacyItems) {System.out.println(pharmacyItem.getCode()); }}}
太好了吗? 运行该课程后,您将看到显示的药房商品代码列表。 这些药房信息通过具有复杂环境的Web应用程序部署的Web服务来获取。
就是本教程中的内容。 希望对您有所帮助,如果您从本教程中学习到一些东西,请不要忘了发表评论。
参考: 如何将现有服务公开为Web服务? 从我们的JCG合作伙伴 Semika loku kaluge在Code Box博客上获得。
翻译自: https://www.javacodegeeks.com/2012/06/expose-java-service-as-web-service.html
java web服务