java实现REST方式的webService

此文章是基于  搭建Jquery+SpringMVC+Spring+Hibernate+MySQL平台

一. 简介 

    WebService有两种方式,一是SOAP方式,二是REST方式。SOAP是基于XML的交互,WSDL也是一个XML文档,可以使用WSDL作为SOAP的描述文件;REST是基于HTTP协议的交互,支持JSON、XML等交互,不需要WSDL。

二. jar包介绍

  1. 点击此下载 apache-cxf-3.0.3,并在 lib 文件夹下得到:

commons-codec-1.9.jar
cxf-core-3.0.3.jar
cxf-rt-frontend-jaxrs-3.0.3.jar
cxf-rt-transports-http-3.0.3.jar
javax.ws.rs-api-2.0.1.jar

  2. commons-httpclient-3.1.jar

三. 相关程序代码

  1. applicationInterface.xml:spring配置文件

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:tx="http://www.springframework.org/schema/tx" xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:jaxrs="http://cxf.apache.org/jaxrs"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.1.xsd
http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.1.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.1.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.1.xsd
http://cxf.apache.org/jaxrs http://cxf.apache.org/schemas/jaxrs.xsd"> <!-- cxf发布rest服务 -->
<import resource="classpath:META-INF/cxf/cxf.xml"/>
<!-- 这里是cxf2.x版本的配置,3.0没有了要去掉 -->
<!-- <import resource="classpath:META-INF/cxf/cxf-extension-soap.xml" /> -->
<import resource="classpath:META-INF/cxf/cxf-servlet.xml" /> <bean id="receiver" class="com.ims.test.webService.cxf.Receiver" /> <jaxrs:server id="serviceContainer" address="/">
<jaxrs:serviceBeans>
<ref bean="receiver" />
</jaxrs:serviceBeans>
</jaxrs:server>
<!-- cxf发布rest服务 --> </beans>

  2. web.xml:web工程配置文件

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:web="http://xmlns.jcp.org/xml/ns/javaee"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_4.xsd
http://xmlns.jcp.org/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
version="2.4"> <context-param>
<param-name>contextConfigLocation</param-name>
<param-value>
classpath:applicationInterface.xml
</param-value>
</context-param> <servlet>
<servlet-name>CXFServlet</servlet-name>
<servlet-class>org.apache.cxf.transport.servlet.CXFServlet</servlet-class>
</servlet> <servlet-mapping>
<servlet-name>CXFServlet</servlet-name>
<url-pattern>/service/*</url-pattern>
</servlet-mapping> </web-app>

  3. 基础服务

    a. BaseService.java:接口

package com.ims.interfaces.webService.cxf;

import javax.ws.rs.Consumes;
import javax.ws.rs.POST;
import javax.ws.rs.Produces;
import javax.ws.rs.QueryParam;
import javax.ws.rs.core.MediaType; public interface BaseService {
@POST
@Consumes(MediaType.TEXT_HTML)
@Produces(MediaType.TEXT_HTML)
public void execute(@QueryParam("type") String type);
}

    b. BaseServiceImpl.java:实现

package com.ims.interfaces.webService.cxf;

import java.io.BufferedInputStream;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.ws.rs.Consumes;
import javax.ws.rs.POST;
import javax.ws.rs.Produces;
import javax.ws.rs.QueryParam;
import javax.ws.rs.core.Context;
import javax.ws.rs.core.MediaType; /**
* 基础服务接口
*/
public abstract class BaseServiceImpl implements BaseService{
@Context
protected HttpServletRequest request;
@Context
protected HttpServletResponse response; @Override
@POST
@Consumes(MediaType.TEXT_HTML)
@Produces(MediaType.TEXT_HTML)
public void execute(@QueryParam("type") String type) {
try {
// 获取请求类型
RequestType reqType = RequestType.getTypeByCode(Integer.valueOf(type)); // 获取HTTP请求体XML字符输入流
BufferedReader reader = new BufferedReader(
new InputStreamReader(
new BufferedInputStream(request.getInputStream()), "gb2312")); // 业务处理
ResponseResult result = process(reqType, reader);
// 设置响应编码
response.setStatus(result.getStatus().getCode()); // 输出响应体
response.setCharacterEncoding("gb2312");
OutputStream outputStream = response.getOutputStream();
outputStream.write(result.getResult().getBytes("gb2312"));
}
catch (IOException ioe) {
response.setStatus(HttpServletResponse.SC_INTERNAL_SERVER_ERROR);
}
} public abstract ResponseResult process(RequestType reqType, BufferedReader reader);
}

  4. Receiver.java:接收端类

package com.ims.test.webService.cxf;

import java.io.BufferedReader;

import javax.ws.rs.Path;

import com.ims.interfaces.webService.cxf.BaseServiceImpl;
import com.ims.interfaces.webService.cxf.RequestType;
import com.ims.interfaces.webService.cxf.ResponseResult;
import com.ims.interfaces.webService.cxf.ResponseStatus; @Path(value = "/receive.do")
public class Receiver extends BaseServiceImpl { @Override
public ResponseResult process(RequestType reqType, BufferedReader reader) {
ResponseResult result = null; switch(reqType){
case UNKNOWN:
result = new ResponseResult(ResponseStatus.CODE400, "");
break;
default:
break;
} return result;
} }

  5. Sender.java:发送端类

package com.ims.test.webService.cxf;

import org.apache.commons.httpclient.HttpClient;
import org.apache.commons.httpclient.HttpStatus;
import org.apache.commons.httpclient.methods.PostMethod;
import org.apache.commons.httpclient.methods.StringRequestEntity; public class Sender { public static void main(String[] args) {
String url = "http://localhost:8090/ims/service/receive.do?type=0";
String xmlString = "<?xml version=\"1.0\" encoding=\"gbk\"?>";
post(url, xmlString);
} /**
* 发送xml数据请求到server端
* @param url xml请求数据地址
* @param xmlString 发送的xml数据流
* @return null发送失败,否则返回响应内容
*/
public static boolean post(String url,String xmlString){
boolean result = false;
//创建httpclient工具对象
HttpClient client = new HttpClient();
//创建post请求方法
PostMethod myPost = new PostMethod(url);
//设置请求超时时间
client.getHttpConnectionManager().getParams().setConnectionTimeout(300*1000); try{
//设置请求头部类型
myPost.setRequestHeader("Content-Type","text/html");
myPost.setRequestEntity(new StringRequestEntity(xmlString,"text/html","gbk"));
int statusCode = client.executeMethod(myPost);
if(statusCode == HttpStatus.SC_OK){
result = true;
}
}catch (Exception e) {
e.printStackTrace();
}finally{
myPost.releaseConnection();
}
return result;
}
}

四. 测试

  eclipse上启动ims工程(webService接收端),以java执行Sender类,eclipse上调试查看Receive类是否收到xml文件

五. 备注

  1. cxf 是支持多线程的,默认的线程池配置可查看 org.apache.cxf.workqueue.AutomaticWorkQueueImpl 源文件:

static final int DEFAULT_MAX_QUEUE_SIZE = 256;

。。。。。。。。。

public AutomaticWorkQueueImpl() {
this(DEFAULT_MAX_QUEUE_SIZE);
}
public AutomaticWorkQueueImpl(String name) {
this(DEFAULT_MAX_QUEUE_SIZE, name);
}
public AutomaticWorkQueueImpl(int max) {
this(max, "default");
}
public AutomaticWorkQueueImpl(int max, String name) {
this(max,
0,
25,
5,
2 * 60 * 1000L,
name);
}
public AutomaticWorkQueueImpl(int mqs,
int initialThreads,
int highWaterMark,
int lowWaterMark,
long dequeueTimeout) {
this(mqs, initialThreads, highWaterMark, lowWaterMark, dequeueTimeout, "default");
}
public AutomaticWorkQueueImpl(int mqs,
int initialThreads,
int highWaterMark,
int lowWaterMark,
long dequeueTimeout,
String name) {
this.maxQueueSize = mqs == -1 ? DEFAULT_MAX_QUEUE_SIZE : mqs;
this.initialThreads = initialThreads;
this.highWaterMark = -1 == highWaterMark ? Integer.MAX_VALUE : highWaterMark;
this.lowWaterMark = -1 == lowWaterMark ? Integer.MAX_VALUE : lowWaterMark;
this.dequeueTimeout = dequeueTimeout;
this.name = name;
this.changeListenerList = new ArrayList<PropertyChangeListener>();
} 。。。。。。。。。 protected synchronized ThreadPoolExecutor getExecutor() {
if (executor == null) {
threadFactory = createThreadFactory(name);
executor = new ThreadPoolExecutor(lowWaterMark,
highWaterMark,
TimeUnit.MILLISECONDS.toMillis(dequeueTimeout),
TimeUnit.MILLISECONDS,
new LinkedBlockingQueue<Runnable>(maxQueueSize),
threadFactory) {
@Override
protected void terminated() {
ThreadFactory f = executor.getThreadFactory();
if (f instanceof AWQThreadFactory) {
((AWQThreadFactory)f).shutdown();
}
if (watchDog != null) {
watchDog.shutdown();
}
}
};

  2. 若要配置线程池参数,可配置 cxf-core-3.0.3.jar\META-INF\cxf\cxf.xml 文件:

<bean id="default-wq" class="org.apache.cxf.workqueue.AutomaticWorkQueueImpl">
<constructor-arg index="0">
<!-- Max number of elements in the queue -->
<value>20000</value>
</constructor-arg>
<constructor-arg index="1">
<!-- name -->
<value>default</value>
</constructor-arg>
<property name="name" value="default" />
<property name="highWaterMark" value="30" />
</bean>
上一篇:tp5服务器验证案例


下一篇:WebService -- Java 实现之 CXF ( 使用:Spring+CXF+Tomcat发布webService)