責(zé)任鏈模式:責(zé)任鏈模式可以用在這樣的場景,當(dāng)一個(gè)request過來的時(shí)候,需要對這個(gè)request做一系列的加工,使用責(zé)任鏈模式可以使每個(gè)加工組件化,減少耦合。也可以使用在當(dāng)一個(gè)request過來的時(shí)候,需要找到合適的加工方式。當(dāng)一個(gè)加工方式不適合這個(gè)request的時(shí)候,傳遞到下一個(gè)加工方法,該加工方式再嘗試對request加工。
????在tomcat中容器之間的調(diào)用使用的就是責(zé)任鏈的設(shè)計(jì)模式,當(dāng)一個(gè)請求過來的時(shí)候首先是engine容器接受請求,然后engine容器會(huì)把請求傳到host容器,host容器又會(huì)傳到context容器,context容器傳到wrapper容器,最后wrapper容器使用適配請求的servlet處理請求。tomcat實(shí)現(xiàn)容器間的責(zé)任傳遞主要涉及到這三個(gè)接口:Container,Pipeline,Valve.看下這三者的關(guān)系,以StandardEngine為例:
?
?
????其中ContainerBase實(shí)現(xiàn)自Container,圖中為了簡潔沒有標(biāo)示出來。
?
????valve中有三個(gè)關(guān)鍵的方法:
public Valve getNext(); // 返回下一個(gè)valve public void setNext(Valve valve); // 設(shè)置下一個(gè)valve public void invoke(Request request, Response response) throws IOException, ServletException; // invoke為實(shí)際要執(zhí)行的代碼
?
?
????Container中有一個(gè)關(guān)鍵方法:
public Pipeline getPipeline(); // 獲取一個(gè)閥門管道
????當(dāng)一個(gè)請求過來的時(shí)候,會(huì)調(diào)用StandardEngine中的getPipeline,然后依次執(zhí)行其中valve;
????初始開始的代碼在CoyteAdapter的407行,如下:
connector.getService().getContainer().getPipeline().getFirst().invoke(request, response);
????connector.getService().getContainer()獲得是的StandardEngine的實(shí)例。
????假設(shè)最后要訪問的servlet為TestServlet可以看到有如下調(diào)用。
?
????
????從StandardEngineValve中會(huì)一直調(diào)用到standardWrapperValve。tomcat的容器之間的調(diào)用就是通過這種方式來調(diào)用的。這樣帶來的好處就是,如果容器中需要新加一個(gè)閥門,只需要?jiǎng)?chuàng)建這個(gè)閥門,并且添加到pipeline中即可,不會(huì)影響到其他閥門代碼,也不需要修改外部調(diào)用容器執(zhí)行pipeline的代碼
更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號(hào)聯(lián)系: 360901061
您的支持是博主寫作最大的動(dòng)力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長非常感激您!手機(jī)微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
