Jetty的Servlt请求路由与ContextHandlerColleection的实现

来源:互联网 发布:红蜘蛛破解软件 编辑:程序博客网 时间:2024/06/07 18:56

们一般会在tomcat下面部署多个应用,每个应用都对应着一个自己的context,那么就需要一个collection将他们管理起来,而且需要对http请求进行路由,将http请求交个相应的应用来处理。。。。

这件事情在jetty中就是ContextHanlerCollection干的事情,它维护者当前jetty中部署的应用。。。将http请求交给匹配的context,然后context再转由内部的servlet来处理。。。。

好了。。还是先来看看ContextHanlerCollection的继承体系吧:



本身继承还是很简单的,前面已经说过了,整个jetty在这方面都还算比较简单一些。。。

其实从名字就能够知道这是一个collection,用于存储handler,其实这里的handler就是前面的文章分析过的contextHandler。。。。


这里我们先来看看HandlerCollection的定义吧。。。这里我们就来看看一些比较重要的内容就好了。。。。

[java] view plaincopy


  1. //添加一个handler,其实是要将其添加到最后面去,说白了就是建立一个数组  
  2. public void addHandler(Handler handler) {  
  3.     setHandlers((Handler[])LazyList.addToArray(getHandlers(), handler, Handler.class));  
  4. }  

这个是添加handler的方法,其实也就是建立一个数组,新添加的handler将会在这个数组的最后面。。。

[java] view plaincopy


  1. //其实就是启动保存的所有的handler  
  2. protected void doStart() throws Exception {  
  3.     MultiException mex=new MultiException();  
  4.     if (_handlers!=null)  
  5.     {  
  6.         for (int i=0;i<_handlers.length;i++)  
  7.             try{_handlers[i].start();}catch(Throwable e){mex.add(e);}  
  8.     }  
  9.     super.doStart();  
  10.     mex.ifExceptionThrow();  
  11. }  

启动,其实也没啥意思,主要就是启动当前所包含的所有的handler

另外handlerCollection本身也有handle方法,不过这个方法的实现比较的扯淡,对于http请求,它会遍历所有的handler,然后直到找到一个handler将这个http请求处理了为止。。。它的实现将会在ContextHandlerCollection中被覆盖。。。


好了,接下来我们来看看ContextHandlerCollection的定义吧,先来看两个比较重要的属性的定义:

[java] view plaincopy


  1. private PathMap _contextMap;  //用于context的map,将contextPath与contextHandler对应起来,而且还有lazy匹配,一些前缀,后缀匹配什么的  
  2. private Class _contextClass = ContextHandler.class;    //这里是用到的handler的类型  

首先前面的contextMap就是用于维护contextPath与contextHandler的对应关系,PathMap是jetty自己定义的类型,它继承自HashMap,但是对其进行了一些扩展,加上了前缀匹配,后缀匹配。。。lazy匹配什么的。。。

至于PathMap的分析留给以后吧,,,其实还算是比较有意思的东西,毕竟所有的http请求都要先到这里经由它来匹配出相应的contextHandler,还算是比较重的。。。。而且在contextHandler中又要用到它来处理path与servlet的对应关系。。。

我们来看看它的doStart方法吧:

[java] view plaincopy


  1. //启动当前的组件  
  2. protected void doStart() throws Exception {  
  3.     mapContexts();  //对当前所有的app的context进行map  
  4.     super.doStart();  //父类的启动组要是启动当前所有的handler  
  5. }  

这里第一个方法的执行是比较重要的,它用于创建pathMap,创建当前所有的handler与path之间的对应关系。。。然后在执行父类的doStart方法,前面已经说过,在父类中主要是进行handler的启动工作。。。

好饿了。。那么我们来看看这个重要的mapContexts方法吧:

[java] view plaincopy


  1. //用于map请求的path,这里会创建PathMap,用于path来索引相应的contextHandler  
  2. public void mapContexts() {  
  3.     PathMap contextMap = new PathMap();  //创建一个pathmap  
  4.     Handler[] branches = getHandlers();  //获取所有的handler  
  5.       
  6.     //遍历当前的所有的handler  
  7.     for (int b=0;branches!=null && b<branches.length;b++)  {  
  8.         Handler[] handlers=null;  
  9.           
  10.         if (branches[b] instanceof ContextHandler) {   //如果是contextHandler需要新创建爱你一个数组  。。?为啥。。?   
  11.             handlers = new Handler[]{ branches[b] };  
  12.         } else if (branches[b] instanceof HandlerContainer) {  
  13.             handlers = ((HandlerContainer)branches[b]).getChildHandlersByClass(ContextHandler.class);  
  14.         } else {  
  15.             continue;  
  16.         }  
  17.         //遍历当前的contextHandler  
  18.         for (int i=0;i<handlers.length;i++) {  
  19.             ContextHandler handler=(ContextHandler)handlers[i];  
  20.   
  21.             String contextPath=handler.getContextPath();  //获取当前这个handler的context的path  
  22.             //判断不合规的contextPath  
  23.             if (contextPath==null || contextPath.indexOf(‘,’)>=0 || contextPath.startsWith(“*”))  
  24.                 throw new IllegalArgumentException (“Illegal context spec:”+contextPath);  
  25.   
  26.             if(!contextPath.startsWith(“/”))  //如果不是/开头的,那么给它加上  
  27.                 contextPath=’/’+contextPath;  
  28.   
  29.             if (contextPath.length()>1) {  
  30.                 if (contextPath.endsWith(“/”))  //表示一个路径下的所有,那么就为其添加*  
  31.                     contextPath+=”*”;  
  32.                 else if (!contextPath.endsWith(“/*”))  
  33.                     contextPath+=”/*”;   //反正都是要变成/*  
  34.             }  
  35.             //也就是最后的contextPath都要变成类似于:/manager/*  类型的  
  36.   
  37.             Object contexts=contextMap.get(contextPath);     //这个path对应的所有contexts,这里可以理解为其实一个数组  
  38.             String[] vhosts=handler.getVirtualHosts();  //获取虚拟主机名  
  39.   
  40.               
  41.             if (vhosts!=null && vhosts.length>0) {  //如果有虚拟主机名字  
  42.                 Map hosts;  
  43.   
  44.                 if (contexts instanceof Map) {  
  45.                     hosts=(Map)contexts;  
  46.                 }  else {  
  47.                     hosts=new HashMap();   
  48.                     hosts.put(”*”,contexts);  
  49.                     contextMap.put(contextPath, hosts);  
  50.                 }  
  51.   
  52.                 for (int j=0;j<vhosts.length;j++)  
  53.                 {  
  54.                     String vhost=vhosts[j];  
  55.                     contexts=hosts.get(vhost);  
  56.                     contexts=LazyList.add(contexts,branches[b]);  
  57.                     hosts.put(vhost,contexts);  
  58.                 }  
  59.             } else if (contexts instanceof Map) {  
  60.                 Map hosts=(Map)contexts;  
  61.                 contexts=hosts.get(”*”);  
  62.                 contexts= LazyList.add(contexts, branches[b]);  
  63.                 hosts.put(”*”,contexts);  
  64.             } else  {  
  65.                 contexts= LazyList.add(contexts, branches[b]);  //为这个contexts添加handler  
  66.                 contextMap.put(contextPath, contexts);  //将这个paht与这个context对应起来,里面会进行前缀和后缀的处理  
  67.             }  
  68.         }  
  69.     }   
  70.     _contextMap=contextMap;  //让当前的contextMap指向刚刚创建的pathMap  
  71.   
  72. }  

代码还挺长的。。。其实主要要做的事情如下:

(1)遍历当前所有的handler,其实也就是contextHandler

(2)预处理他们的path,一般情况下例如加入我们的工程为manager,那么部署在jetty,默认的path将是/manager,这里的预处理是将其变成/manager/*这个样子。。。

(3)将其放到pathMap里面去。。。让pathMap来处理对应关系。。。有前缀,后缀匹配什么的。。这里还涉及到虚拟主机名什么的。。就不细看了。。


最后我们再来看看另外一个很重要的方法,handle方法,ContextHandlerCollection的handle方法的用处就是将http请求分发给对应的contextHandler来处理。。。

来看他的定义吧:

[java] view plaincopy


  1.   //这里的target是没有经过处理的,直接就是最原始的url后面的那些,例如/manager/public/aa.jpg,也就是包括了前面的contextPath  
  2.   //这个函数用于将http请求路由给匹配的contextHandler来处理  
  3.   public void handle(String target, HttpServletRequest request, HttpServletResponse response, int dispatch) throws IOException, ServletException {  
  4.       Handler[] handlers = getHandlers();   //获得当前所有的handler  
  5.       if (handlers==null || handlers.length==0)  
  6.         return;  
  7.   
  8.       Request base_request = HttpConnection.getCurrentConnection().getRequest();  //获取当前的http请求,前面会将它保存在线程变量中  
  9.       
  10.       // data structure which maps a request to a context  
  11.       // each match is called in turn until the request is handled  
  12.       // { context path =>   
  13.       //     { virtual host => context }   
  14.       // }  
  15. PathMap map = _contextMap;  //当前的context的map  
  16. if (map!=null && target!=null && target.startsWith(“/”)) {  
  17.     //这里获取所有匹配的context  
  18.     Object contexts = map.getLazyMatches(target);  //获取匹配的context  
  19.           for (int i=0; i<LazyList.size(contexts); i++) {  //遍历当前的context  
  20.               // then, match against the virtualhost of each context  
  21.               Map.Entry entry = (Map.Entry)LazyList.get(contexts, i);  
  22.               Object list = entry.getValue();  
  23.   
  24.               if (list instanceof Map) {  
  25.                   Map hosts = (Map)list;  
  26.                   String host = normalizeHostname(request.getServerName());  
  27.   
  28.                   // explicitly-defined virtual hosts, most specific  
  29.                   list=hosts.get(host);  
  30.                   for (int j=0; j<LazyList.size(list); j++) {  
  31.                       Handler handler = (Handler)LazyList.get(list,j);  
  32.                       handler.handle(target,request, response, dispatch);  
  33.                       if (base_request.isHandled())  
  34.                           return;  
  35.                   }  
  36.   
  37.                   // wildcard for one level of names   
  38.                   list=hosts.get(”*.”+host.substring(host.indexOf(“.”)+1));  
  39.                   for (int j=0; j<LazyList.size(list); j++)  
  40.                   {  
  41.                       Handler handler = (Handler)LazyList.get(list,j);  
  42.                       handler.handle(target,request, response, dispatch);  
  43.                       if (base_request.isHandled())  
  44.                           return;  
  45.                   }  
  46.   
  47.                   // no virtualhosts defined for the context, least specific  
  48.                   // will handle any request that does not match to a specific virtual host above                      
  49.                   list=hosts.get(”*”);  
  50.                   for (int j=0; j<LazyList.size(list); j++)  
  51.                   {  
  52.                       Handler handler = (Handler)LazyList.get(list,j);  
  53.                       handler.handle(target,request, response, dispatch);  
  54.                       if (base_request.isHandled())  
  55.                           return;  
  56.                   }  
  57.               } else {  
  58.                 //一般都是执行这里,因为前面获取所有匹配的context的时候会将所有匹配上的handler构成一个数组  
  59.                   for (int j=0; j<LazyList.size(list); j++) {  
  60.                       Handler handler = (Handler)LazyList.get(list,j);  
  61.                       handler.handle(target,request, response, dispatch);  
  62.                       if (base_request.isHandled())  
  63.                           return;  //如果有handler处理了,那么就不处理了  
  64.                   }  
  65.               }  
  66.     }  
  67. else {  
  68.            // This may not work in all circumstances… but then I think it should never be called  
  69.     for (int i=0;i<handlers.length;i++) {  
  70.         handlers[i].handle(target,request, response, dispatch);  
  71.         if ( base_request.isHandled())  
  72.             return;  
  73.     }  
  74. }  
  75.   }  

因为这里有了一些虚拟主机什么的东西,所以看起来稍微复杂一些。。但是其实最主要做的事情如下:

(1)调用当前contextMap的getLazyMatches方法,找到所有与当前匹配的contextHandle

(2)遍历这些handler,然后依次调用他们的handle方法来处理这个请求,知道这次的请求被处理了为止。。



好了,到这里ContextHandlerCollection的主要类容就算看完了。。。它还算是比较重要的。。。其实到这里为止http请求在jetty中的处理流程就算是比较的清楚了。。。用一张图来说明一下吧:



0 0
原创粉丝点击