`
ajax
  • 浏览: 251821 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

Restlet实战(三十)(完结篇)运行流程之源代码分析(续)

    博客分类:
  • REST
阅读更多

前面一篇文章分析了servlet里init方法,包括init方法本身以及调用的方法源代码,这篇文章继续,按照servlet的生命周期,接下去会调用servlet的service方法:

 

@Override
public void service(HttpServletRequest request, HttpServletResponse response)
		throws ServletException, IOException {
	final HttpServerHelper helper = getServer(request);

	if (helper != null) {
		helper.handle(createCall(helper.getHelped(), request, response));
	} else {
		log("[Noelios Restlet Engine] - Unable to get the Restlet HTTP server connector. Status code 500 returned.");
		response.sendError(500);
	}
}

 

让我们一步一步的分析下去,首先看getServer方法:

 

public HttpServerHelper getServer(HttpServletRequest request) {
	HttpServerHelper result = this.helper;

	if (result == null) {
		synchronized (ServerServlet.class) {
			if (result == null) {
				// Find the attribute name to use to store the server
				// reference
				final String serverAttributeName = getInitParameter(
						NAME_SERVER_ATTRIBUTE,
						NAME_SERVER_ATTRIBUTE_DEFAULT);

				// Look up the attribute for a target
				result = (HttpServerHelper) getServletContext()
						.getAttribute(serverAttributeName);

				if (result == null) {
					result = createServer(request);
					getServletContext().setAttribute(serverAttributeName,
							result);
				}

				this.helper = result;
			}
		}
	}

	return result;
}

 

上面这段代码首先还是根据server属性名字,查找servlet context里是否已经有存放的server,如果有就直接取出来,如果没有,就创建一个server,然后把这个新的server放到servlet context里,以便下次的时候不需要重新创建,这个动作之前的component已经重复过。如果第一次请求,肯定是没有server存在,那么接下来,看看是怎么创建一个server的。由于这个方法的代码很长,所以,只贴出部分代码做一个解释:

 

在这个方法里,首先得到Component,调用的getComponent,之前文章的分析,我们知道,这个时候已经有component放到servlet context。

 

final Component component = getComponent();

 

既然有了component,那么接下去就会创建一个server以及一个server helper,需要说明的是restlet代码里有很多helper,如ComponentHelper,ApplicationHelper,HttpServerHelper等等。

 

final Server server = new Server(component.getContext()
		.createChildContext(), (List<Protocol>) null, request
		.getLocalAddr(), request.getLocalPort(), component);
result = new HttpServerHelper(server);

 

然后,获取请求的path,如/restlet/resources/customers/1:

 

final String uriPattern = request.getContextPath()
                    + request.getServletPath();

 

其中restlet是context path,而/resources是servlet path,而/customers/1则是我们定义的映射的URI。

 

接下来会判断当前的Component是否是default component(关于default component的判断,在上篇文章中解释过),如果是default compoent就将之前在init方法里创建的application attach到当前component的默认的Host上(虚拟主机)。否则,就会启动另外的一个流程:

 

1.首先会判断一个参数org.restlet.autoWire是否是true,那么这个参数代表什么呢?看看源码里面的注释:

 

 

org.restlet.autoWire是Servlet context的一个参数名字,并包含一个布尔型值,值为true表示所有的applications
将被用Servlet上下文的路径值attach到Component的虚拟主机(virtual hosts)上

 

代码如下:

 

final String autoWire = getInitParameter(AUTO_WIRE_KEY,
		AUTO_WIRE_KEY_DEFAULT);
if (AUTO_WIRE_KEY_DEFAULT.equalsIgnoreCase(autoWire)) {

 

根据我在开始时候列出的环境,没有在web.xml中配置一个名为org.restlet.autoWire的参数,所以,这里autoWire会取默认值为true。而常量AUTO_WIRE_KEY_DEFAULT的值也为true。

 

2.分析映射的URL是否需要增加Context path或者Servlet path

 

  源码里面定义了两个变量:

 

boolean addContextPath = false;
boolean addFullServletPath = false;

 

顾名思义,第一个是判断是否需要增加context path,第二个是判断是否需要增加servlet path。下面的代码就是判断的具体逻辑:

 

for (final Route route : component.getDefaultHost()
                                .getRoutes()) {
	if (route.getTemplate().getPattern() == null) {
		addFullServletPath = true;
		continue;
	}

	if (!route.getTemplate().getPattern().startsWith(
			uriPattern)) {
		if (!route.getTemplate().getPattern()
				.startsWith(request.getServletPath())) {
			addFullServletPath = true;
		} else {
			addContextPath = true;
			break;
		}
	}
}

 

代码比较简单,不多做解释,接下来,如果对于默认主机的定义的所有路由的URL不需要增加Context path,则会继续检查attach到该component的所有的别的虚拟主机是否需要增加Context path和Servlet path:

 

 

if (!addContextPath) {
	for (final VirtualHost virtualHost : component.getHosts()) {
		if (virtualHost.getRoutes().isEmpty()) {
			// Case where the default host has a default
			// route (with an empty pattern).
			addFullServletPath = virtualHost .getDefaultRoute() != null;
		} else {
			for (final Route route : virtualHost.getRoutes()) {
				if (route.getTemplate().getPattern() == null) {
					addFullServletPath = true;
					continue;
				}
				if (!route.getTemplate().getPattern().startsWith(uriPattern)) {
					if (!route.getTemplate().getPattern().startsWith(request.getServletPath())) {
						addFullServletPath = true;
					} else {
						addContextPath = true;
						break;
					}
				}
			}
		}
		if (addContextPath) {
			break;
		}
	}
}

 

 3. 改变各种路由

 

经过上面两步,如果需要增加Context path或者Servlet path,则会判断是那种类型:

 

if (addContextPath) {
	offsetPath = request.getContextPath();
} else {
	offsetPath = uriPattern;
}

 

接下来会对所有的路由进行转化,包括默认的虚拟主机的默认路由,默认虚拟主机的路由,组件的别的虚拟主机的默认路由以及组件的别的虚拟主机的路由。这里为了节省篇幅,仅给出一种情况的代码:

 

// Shift the routes of the default host
for (final Route route : component.getDefaultHost()
		.getRoutes()) {
	log("[Noelios Restlet Engine] - Attaching restlet: "
			+ route.getNext()
			+ " to URI: "
			+ offsetPath
			+ route.getTemplate().getPattern());
	route.getTemplate().setPattern(
			offsetPath
					+ route.getTemplate().getPattern());
}

 

ok,分析完这几段代码后,结合我之前给出的环境,因为定义url映射时候,没有Context path和servlet path,只是单纯的映射部分,如/customers/1,而不是/restlet/resources/customers/1或者/resources/customers/1.所以,经过上述代码,我们的定义的url会变成/{Context path}/{Servlet path}/{Mapping url}.

 

至此,一个Server就创建完成,然后系统会根据一个属性名字将它放到一个Servlet Context里。然后返回基于Server的一个Helper。

 

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics