需求
公司有一个产品,包括前台WEB界面和多个后台服务,各个服务都需要在前面界面中进行配置和控制,以调整服务的行为。以前,配置文件都存放在数据库中,界面上修改配置后入库,并发送消息(Socket)通知特定的服务重新加载配置。这样有些问题,一方面自己维护Socket带来很多麻烦,二来数据库重建的时候,需要备份/恢复这些配置数据。
所以,我们想把配置文件局部化到各个服务(比如用本地文件存储),然后在界面上修改的时候,实时向服务请求当前配置数据(XML格式),修改完毕后,再直接发给服务进行更新和存储。而在通信方式上,我们希望各个服务提供Web Service接口来实现配置的检索和更新。
但Web Service通常需要在Web Container(比如,tomcat, jboss)中实现,而我们不想把所有的服务都跑在tomcat里,于是想找到一种更加轻量级的方式。
今天偶然看到JSR311,进而发现在Java后台服务中内嵌grizzly(基于NIO的实现)和Jersey(Sun的JSR311参考实现)来提供Restful Web Service能力,是非常方便的。个人认为这种方式适用于后台Java程序的控制、配置和监视,其作用有些类似于JMX,但比实现JMX要简单的多(JSR311基于POJO)。
背景知识
Representational state transfer (REST) Web Service:
它是Roy Fielding博士在2000年他的博士论文中提出来的一种软件架构风格。在目前三种主流的Web Service实现方案中,因为REST与SOAP和XML-RPC相比显的更加简洁,因此越来越多的Web Service开始采用REST风格设计和实现。
参考资料:http://en.wikipedia.org/wiki/Representational_State_Transfer
Jersey:
它是Sun对JSR311的官方参考实现,而JSR311是java中实现Restful Web Service的API规范(JSR311: JAX-RS: The Java API for RESTful Web Services)。JSR311有一个重要目标:使用注解(annotation)把POJO暴露成Web Service,这样就比较轻量级。
参考资料:https://jsr311.dev.java.net/nonav/releases/1.0/spec/index.html
Grizzly:
Grizzly于2004年诞生在GlassFish中,开始目的是要建构一个HTTP Web服务器,用来代替Tomcat的Coyote连接器和Sun WebServ er6.1。后来,Grizzly成为一种应用程序框架,专门解决编写成千上万用户访问服务器时候产生的各种问题。使用JAVA NIO作为基础,并隐藏其编程的复杂性。在本例中,我们将其用作内嵌的Servlet Container。
参考资料:https://grizzly.dev.java.net/
准备工作
首先,下载grizzly和jersey。其中,grizzly的下载地址为:http://download.java.net/maven/2/com/sun/grizzly/grizzly-servlet-webserver/1.9.18a/grizzly-servlet-webserver-1.9.18a.jar,jersey的下载地址为:http://download.java.net/maven/2/com/sun/jersey/jersey-archive/1.1.2-ea/jersey-archive-1.1.2-ea.zip
在Eclipse中建一个Java工程,名为jsr331,然后把下载的jersey-archive-1.1.2-ea.zip解压,将jersey-archive-1.1.2-ea\contribs、jersey-archive-1.1.2-ea\lib两个目录下的jar包,连同下载的grizzly-http-webserver-1.9.18a.jar都拷贝到jsr331工程下的lib目录,并加入到该工程的Build Path。(实测中,发现还需要引入一个包,下载地址:http://repository.jboss.org/maven2/org/jvnet/mimepull/1.2/mimepull-1.2.jar
上面这个连接不对,我来修正下:
http://download.java.net/maven/2/org/jvnet/mimepull/1.2/
)
编写最简单的服务
JSR331把准备提供Web Service的类称为Resource class。Resource class是一个普通类(POJO),但是按照规范要求增加了特定的注解(annotation)。我们首先实现最简单的hello world服务。
在jinxfei.test.jsr311.service下,建立HelloService类,内容如下:
代码中的注解(annotation)决定了程序发布成Web Service后的行为和特性。其中,HelloService类前面的@PATH,表明该Service的URL路径,这种类名前面带@PATH注解的类被称为Root Resource Class,因为他们决定了访问Service时URI中的第一级路径;@GET表示访问该服务使用HTTP GET方法;@Produces规定该服务返回结果的类型,这里的”text/plain”表名返回纯文本。
发布服务
实现服务类后,我们要启动一个内嵌的grizzly servlet container,并把HelloService发布到该Container中,这样就能通过HTTP协议访问该服务。Jersey提供了两种发布方式,标准的做法需要在web.xml中做配置,所以比较适用于部署在独立的Web Container下的应用,本文不做深入介绍,可参见:http://docs.sun.com/app/docs/doc/820-4867/6nga7f5o4?l=en&a=view。本例与grizzly整合,所以使用grizzly的工具类,通过代码进行部署配置。
在jinxfei.test.jsr311包下创建ServiceDeployer类,内容如下:
代码很简单,其中:initParams.put("com.sun.jersey.config.property.packages" , "jinxfei.test.jsr311.service"); 这一行的第二个参数表明服务实现类所在的包名,系统会自动搜索做了注解的类并将其发布成服务。
在Eclipse中运行该类,控制台上打印“服务已启动”字样,表示系统启动成功,打开浏览器,输入:http://localhost:9876/hello,即可看到效果,如下图所示:
让功能再强一点点
接下来,我们要让服务能够接受参数,并根据参数打印不同的信息,最简单的就是回显(Echo)。为了实现这一功能,我们给HelloService类增加一个方法:
-
@POST @Path("echo")
-
@Consumes("application/x-www-form-urlencoded")
-
public String echo(@FormParam("msg") String message){
-
return "Are you saying:"+message;
-
}
@POST表明该方法要用HTTP Post来访问,@Path表明访问该方法的相对路径是echo,@Consumes表明该方法处理HTTP Post请求中何种类型的数据。该方法参数中的注解@FormParam("msg")说明后面的“String message”参数取自表单提交数据中的msg。
由于该方法需要Post访问,且需要通过表单提交数据,所以我们创建一个test.htm,在其中写一个简单的Form:
-
<form action="http://localhost:9876/hello/echo" method="post">
-
<input type="text" name="msg">
-
<input type="submit" value="submit">
-
</form>
重启ServiceDeployer,然后在浏览器中打开test.htm,表单中输入任意信息:
然后点提交按钮,可以看到如下信息:
这说明HelloService已经提取了表单参数,并回显给用户。
-
package jinxfei.test.jsr311;
-
-
import java.io.IOException;
-
import java.net.URI;
-
import java.util.HashMap;
-
import java.util.Map;
-
import javax.ws.rs.core.UriBuilder;
-
import com.sun.grizzly.http.SelectorThread;
-
import com.sun.jersey.api.container.grizzly.GrizzlyWebContainerFactory;
-
-
public class ServiceDeployer {
-
-
public static void main(String[] args) throws IOException {
-
URI ServerURI=UriBuilder.fromUri("http://localhost/").port(9876).build();
-
startServer(ServerURI);
-
System.out.println("服务已启动,请访问:"+ServerURI);
-
}
-
-
protected static SelectorThread startServer(URI serverURI) throws IOException {
-
final Map<String, String> initParams = new HashMap<String, String>();
-
initParams.put("com.sun.jersey.config.property.packages","jinxfei.test.jsr311.service");
-
System.out.println("Grizzly 启动中...");
-
SelectorThread threadSelector = GrizzlyWebContainerFactory.create(serverURI, initParams);
-
return threadSelector;
-
}
-
}
-
package jinxfei.test.jsr311.service;
-
-
import javax.ws.rs.GET;
-
import javax.ws.rs.Path;
-
import javax.ws.rs.Produces;
-
-
@Path("/hello")
-
public class HelloService {
-
@GET
-
@Produces("text/plain")
-
public String helloWorld(){
-
return "Hello world!";
-
}
-
}
分享到:
相关推荐
### Jersey和Tomcat构建RESTful WebService #### RESTful Web服务简介 REST(Representational State Transfer)是由HTTP规范1.0和1.1版的主要作者之一Roy Fielding在其2000年的博士论文中提出的。REST的核心思想...
### Java的WebService支持 #### SOA (Service-Oriented Architecture) 面向服务架构简介 面向服务架构(SOA)是一种软件设计模式,它强调通过标准化的服务接口将应用程序的不同功能单元进行解耦合,以便这些服务能够...
Apache CXF是一个开源的Web服务框架,它支持多种Web服务标准,如SOAP、RESTful API和WS-*规范。CXF提供了丰富的API和工具,帮助开发者构建和部署服务。 2. **Spring 3框架**: Spring是Java企业级应用程序的流行...
下面是一个简单的服务端实现示例,展示了如何使用CXF发布一个WebService: ```java package net.ilkj.soap; import javax.jws.WebMethod; import javax.jws.WebService; import javax.xml.ws.Endpoint; @...
它支持SOAP、RESTful等多种Web服务标准,并且提供了丰富的API和工具,使得开发者能够方便地创建、部署和管理Web服务。本示例代码是基于Eclipse集成开发环境的一个项目,旨在帮助开发者理解和实践如何在Java中使用CXF...
#### 三、CXF WebService入门实例 接下来,我们将通过一个简单的“Hello World”示例来介绍如何使用CXF创建WebService。 **所需jar包**:需要包含CXF的核心jar包以及相关的依赖。 **服务端代码**: ```java ...
在上面的配置中,`<cxf:features>`元素启用了JSON支持,`<cxf:jsr311/>`则支持JAX-RS标准,使得CXF可以处理RESTful请求。`<jaxws:endpoint>`定义了服务的实现类和访问地址。 然后,创建一个Java类(如`...
【CXF 开发实例详解】 CXF(CXF: Apache CXF)是一个开源的Java框架,主要用于构建和服务导向架构(SOA)。它提供了多种服务框架,包括Web服务和RESTful API,同时也支持JAX-WS和JAX-RS标准。CXF以其易用性和丰富的...
在提供的"WebService-master"压缩包中,可能包含示例代码、教程文档或者完整的项目实例,这将有助于进一步学习和掌握WebService的相关技术。通过阅读源码、运行示例,可以加深对WebService工作原理的理解,并提升...
- **REST支持**:Mule 3.0通过集成Jersey框架,提供了本地化的REST和JAX-RS支持,这使得RESTful API的开发变得更加容易。 - **AJAX支持**:Mule 3.0现在直接支持与JavaScript应用程序的集成,可以通过服务端和...
- **RESTful 提供者端点**:讲解了如何使用 JAX-WS 实现 RESTful 风格的服务。 - **提供者与处理器**:说明了如何将处理器与提供者结合使用。 ##### 异步提供者(AsyncProvider) - **异步提供者示例**:通过实例...