Web服务其本质上还是HTTP请求,只不过要遵循Web服务特定的规范,比如说服务端与客户端双方通信的媒介为XML文件格式,以完成对一次Web服务调用的描述。当然此XML还有特定的格式,至于这个特定的格式是什么样的取决于使用哪一种数据绑定方式。CXF中称这为数据绑定,个人觉得称之为Java对象的序列化与反序列化更易懂些。
在这个方法中,完成了对Message的创建,初始化设置(setupMessage);Exchange、Session的创建,并将Message与Session放置进Exchange对象中。Message对象的初始化设置(setupMessage方法中)非常重要,向Message中放置了很多信息,如Http请求对象,响应对象,请求的地址等一系列信息,具体请查看setupMessage方法,需要的对象创建完成后调用ChainInitiationObserver.onMessage()方法,源码如下:
本文将从源码(CXF版本为2.7.6)层面来分析一下CXF是如何完成对一个Web服务(SOAP类型为例)请求的处理的,注意这里分析的是请求的处理,并不包含这个请求是如何生成的,分析将将从CXF服务端接收到一个请求开始。在CXF中,处理HTTP请求是使用Jetty实现的,其处理请求的Handler为org.apache.cxf.transport.http_jetty.JettyHTTPHandler在其handle方法中调用的是org.apache.cxf.transport.http_jetty.JettyHTTPDestination的doService()方法,其中最重要的一句代码是调用了serviceRequest(context, req, resp)方法,下面是源码:
protected void serviceRequest(final ServletContext context,
final HttpServletRequest req,
final HttpServletResponse resp) throws IOException {
Request baseRequest = (req instanceof Request) ? (Request)req : getCurrentRequest();
if (LOG.isLoggable(Level.FINE)) {
LOG.fine("Service http request on thread: " + Thread.currentThread());
}
Message inMessage = retrieveFromContinuation(req);
if (inMessage == null) {
inMessage = new MessageImpl();
ExchangeImpl exchange = new ExchangeImpl();
exchange.setInMessage(inMessage);
setupMessage(inMessage, context, req, resp);
((MessageImpl)inMessage).setDestination(this);
exchange.setSession(new HTTPSession(req));
}
try {
//处理消息
//incomingObserver的现实类为org.apache.cxf.transport.ChainInitiationObserver
incomingObserver.onMessage(inMessage);
resp.flushBuffer();
baseRequest.setHandled(true);
} catch (SuspendedInvocationException ex) {
//省略...
}
//省略...
}
在这个方法中,完成了对Message的创建,初始化设置(setupMessage);Exchange、Session的创建,并将Message与Session放置进Exchange对象中。Message对象的初始化设置(setupMessage方法中)非常重要,向Message中放置了很多信息,如Http请求对象,响应对象,请求的地址等一系列信息,具体请查看setupMessage方法,需要的对象创建完成后调用ChainInitiationObserver.onMessage()方法,源码如下:
public void onMessage(Message m) {
Bus origBus = BusFactory.getAndSetThreadDefaultBus(bus);
ClassLoaderHolder origLoader = null;
try {
if (loader != null) {
origLoader = ClassLoaderUtils.setThreadContextClassloader(loader);
}
InterceptorCha