Spring MVC 核心架構圖html
架構圖對應的DispatcherServlet核心代碼以下:前端
//前端控制器分派方法 protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception { HttpServletRequest processedRequest = request; HandlerExecutionChain mappedHandler = null; int interceptorIndex = -1; try { ModelAndView mv; boolean errorView = false; try { //檢查是不是請求是不是multipart(如文件上傳),若是是將經過MultipartResolver解析 processedRequest = checkMultipart(request); //步驟二、請求處處理器(頁面控制器)的映射,經過HandlerMapping進行映射 mappedHandler = getHandler(processedRequest, false); if (mappedHandler == null || mappedHandler.getHandler() == null) { noHandlerFound(processedRequest, response); return; } //步驟三、處理器適配,即將咱們的處理器包裝成相應的適配器(從而支持多種類型的處理器) HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler()); // 304 Not Modified緩存支持 //此處省略具體代碼 // 執行處理器相關的攔截器的預處理(HandlerInterceptor.preHandle) //此處省略具體代碼 // 步驟四、由適配器執行處理器(調用處理器相應功能處理方法) mv = ha.handle(processedRequest, response, mappedHandler.getHandler()); // Do we need view name translation? if (mv != null && !mv.hasView()) { mv.setViewName(getDefaultViewName(request)); } // 執行處理器相關的攔截器的後處理(HandlerInterceptor.postHandle) //此處省略具體代碼 } catch (ModelAndViewDefiningException ex) { logger.debug("ModelAndViewDefiningException encountered", ex); mv = ex.getModelAndView(); } catch (Exception ex) { Object handler = (mappedHandler != null ? mappedHandler.getHandler() : null); mv = processHandlerException(processedRequest, response, handler, ex); errorView = (mv != null); } //步驟5 步驟六、解析視圖並進行視圖的渲染 //步驟5 由ViewResolver解析View(viewResolver.resolveViewName(viewName, locale)) //步驟6 視圖在渲染時會把Model傳入(view.render(mv.getModelInternal(), request, response);) if (mv != null && !mv.wasCleared()) { render(mv, processedRequest, response); if (errorView) { WebUtils.clearErrorRequestAttributes(request); } } else { if (logger.isDebugEnabled()) { logger.debug("Null ModelAndView returned to DispatcherServlet with name '" + getServletName() + "': assuming HandlerAdapter completed request handling"); } } // 執行處理器相關的攔截器的完成後處理(HandlerInterceptor.afterCompletion) //此處省略具體代碼 catch (Exception ex) { // Trigger after-completion for thrown exception. triggerAfterCompletion(mappedHandler, interceptorIndex, processedRequest, response, ex); throw ex; } catch (Error err) { ServletException ex = new NestedServletException("Handler processing failed", err); // Trigger after-completion for thrown exception. triggerAfterCompletion(mappedHandler, interceptorIndex, processedRequest, response, ex); throw ex; } finally { // Clean up any resources used by a multipart request. if (processedRequest != request) { cleanupMultipart(processedRequest); } } }
核心架構的具體流程步驟以下:node
一、 首先用戶發送請求——>DispatcherServlet,前端控制器收到請求後本身不進行處理,而是委託給其餘的解析器進行處理,做爲統一訪問點,進行全局的流程控制;web
二、 DispatcherServlet——>HandlerMapping, HandlerMapping將會把請求映射爲HandlerExecutionChain對象(包含一個Handler處理器(頁面控制器)對象、多個HandlerInterceptor攔截器)對象,經過這種策略模式,很容易添加新的映射策略;算法
三、 DispatcherServlet——>HandlerAdapter,HandlerAdapter將會把處理器包裝爲適配器,從而支持多種類型的處理器,即適配器設計模式的應用,從而很容易支持不少類型的處理器;數據庫
四、 HandlerAdapter——>處理器功能處理方法的調用,HandlerAdapter將會根據適配的結果調用真正的處理器的功能處理方法,完成功能處理;並返回一個ModelAndView對象(包含模型數據、邏輯視圖名);設計模式
五、 ModelAndView的邏輯視圖名——> ViewResolver, ViewResolver將把邏輯視圖名解析爲具體的View,經過這種策略模式,很容易更換其餘視圖技術;緩存
六、 View——>渲染,View會根據傳進來的Model模型數據進行渲染,此處的Model實際是一個Map數據結構,所以很容易支持其餘視圖技術;服務器
七、返回控制權給DispatcherServlet,由DispatcherServlet返回響應給用戶,到此一個流程結束。網絡
此處咱們只是講了核心流程,沒有考慮攔截器、本地解析、文件上傳解析等,後邊再細述。
近段時間以來,經過接觸有關海量數據處理和搜索引擎的諸多技術,經常見識到很多精妙絕倫的架構圖。除了往往感嘆於每幅圖表面上的繪製的精細以外,更爲架構 圖背後所隱藏的設計思想所歎服。我的這兩天一直在蒐集各大型網站的架構設計圖,一爲了一飽眼福,領略各種大型網站架構設計的精彩以外,二來也可供閒時反覆 琢磨體會,何樂而不爲呢?特此,總結整理了諸如國外wikipedia,Facebook,Yahoo!,YouTube,MySpace,Twitter,國內如優酷網等大型網站的技術架構(本文重點分析優酷網的技術架構),以饗讀者。
本文着重凸顯每一幅圖的精彩之處與其背後含義,而圖的說明性文字則從簡從略。ok,好好享受此番架構盛宴吧。固然,如有任何建議或問題,歡迎不吝指正。謝謝。
WikiPedia 技術架構圖Copy @Mark Bergsma
。
Facebook 搜索功能的架構示意圖
細心的讀者必定能發現,上副架構圖以前出如今此文之中:從幾幅架構圖中偷得半點海里數據處理經驗。本文與前文最大的不一樣是,前文只有幾幅,此文系列將有上百幅架構圖,任您盡情觀賞。
Yahoo! Mail 架構
Yahoo! Mail 架構部署了 Oracle RAC,用來存儲 Mail 服務相關的 Meta 數據。
twitter的總體架構設計圖
twitter平臺大體由twitter.com、手機以及第三方應用構成,以下圖所示(其中流量主要以手機和第三方爲主要來源):
緩存在大型web項目中起到了舉足輕重的做用,畢竟數據越靠近CPU存取速度越快。下圖是twitter的緩存架構圖:
關於緩存系統,還能夠看看下幅圖:
GAE的架構圖
簡單而言,上述GAE的架構分爲如圖所示的三個部分:前端,Datastore和服務羣。
Datastore是基於BigTable技術的分佈式數據庫,雖然其也能夠被理解成爲一個服務,可是因爲其是整個App Engine惟一存儲持久化數據的地方,因此其是App Engine中一個很是核心的模塊。其具體細節將在下篇和你們討論。
整個服務羣包括不少服務供App Server調用,好比Memcache,圖形,用戶,URL抓取和任務隊列等。
Amazon的Dynamo Key-Value存儲架構圖
可能有讀者並不熟悉Amazon,它如今已是全球商品品種最多的網上零售商和全球第2大互聯網公司。而以前它僅僅是一個小小的網上書店。ok,下面,我們來見識下它的架構。
Dynamo是亞馬遜的key-value模式的存儲平臺,可用性和擴展性都很好,性能也不錯:讀寫訪問中99.9%的響應時間都在300ms內。按分佈 式系統經常使用的哈希算法切分數據,分放在不一樣的node上。Read操做時,也是根據key的哈希值尋找對應的node。Dynamo使用了 Consistent Hashing算法,node對應的再也不是一個肯定的hash值,而是一個hash值範圍,key的hash值落在這個範圍內,則順時針沿ring找,碰 到的第一個node即爲所需。
Dynamo對Consistent Hashing算法的改進在於:它放在環上做爲一個node的是一組機器(而不是memcached把一臺機器做爲node),這一組機器是經過同步機制保證數據一致的。
下圖是分佈式存儲系統的示意圖,讀者可觀摩之:
Amazon的雲架構圖以下:
Amazon的雲架構圖
從一開始,優酷網就自建了一套CMS來解決前端的頁面顯示,各個模塊之間分離得比較恰當,前端可擴展性很好,UI的分離,讓開發與維護變得十分簡單和靈活,下圖是優酷前端的模塊調用關係:
這樣,就根據module、method及params來肯定調用相對獨立的模塊,顯得很是簡潔。下圖是優酷的前端局部架構圖:
優酷的數據庫架構也是經歷了許多波折,從一開始的單臺MySQL服務器(Just Running)到簡單的MySQL主從複製、SSD優化、垂直分庫、水平sharding分庫。
其主從複製的過程以下圖所示:
可是,主從複製也帶來其餘一系列性能瓶頸問題:
那問題產生總得解決的,這就產生下面的優化方案。
MySQL垂直分區
若是把業務切割得足夠獨立,那把不一樣業務的數據放到不一樣的數據庫服務器將是一個不錯的方案,並且萬一其中一個業務崩潰了也不會影響其餘業務的正常進行,而且也起到了負載分流的做用,大大提高了數據庫的吞吐能力。通過垂直分區後的數據庫架構圖以下:
然而,儘管業務之間已經足夠獨立了,可是有些業務之間或多或少總會有點聯繫,如用戶,基本上都會和每一個業務相關聯,何況這種分區方式,也不能解決單張表數據量暴漲的問題,所以爲什麼不試試水平sharding呢?
MySQL水平分片(Sharding)
這是一個很是好的思路,將用戶按必定規則(按id哈希)分組,並把該組用戶的數據存儲到一個數據庫分片中,即一個sharding,這樣隨着用戶數量的增長,只要簡單地配置一臺服務器便可,原理圖以下:
如何來肯定某個用戶所在的shard呢,能夠建一張用戶和shard對應的數據表,每次請求先從這張表找用戶的shard id,再從對應shard中查詢相關數據,以下圖所示: 可是,優酷是如何解決跨shard的查詢呢,這個是個難點,據介紹優酷是儘可能不跨shard查詢,實在不行經過多維分片索引、分佈式搜索引擎,下策是分佈式數據庫查詢(這個很是麻煩並且耗性能)。
緩存策略
貌似大的系統都對「緩存」情有獨鍾,從http緩存到memcached內存數據緩存,但優酷表示沒有用內存緩存,理由以下:
並且Squid 的 write() 用戶進程空間有消耗,Lighttpd 1.5 的 AIO(異步I/O) 讀取文件到用戶內存致使效率也比較低下。
但 爲什麼咱們訪問優酷會如此流暢,與土豆相比優酷的視頻加載速度略勝一籌?這個要歸功於優酷創建的比較完善的內容分發網絡(CDN),它經過多種方式保證分佈 在全國各地的用戶進行就近訪問——用戶點擊視頻請求後,優酷網將根據用戶所處地區位置,將離用戶最近、服務情況最好的視頻服務器地址傳送給用戶,從而保證 用戶能夠獲得快速的視頻體驗。這就是CDN帶來的優點,就近訪問。
附註:一、此段優酷網的技術架構整理於此處:http://www.itivy.com/ivy/archive/2011/8/13/the-architecture-of-youku.html;二、同時推薦一個很是好的站點:http://www.dbanotes.net/)。從上百幅架構圖中學得半點大型網站建設經驗(上),完。
此篇文章終於寫完了,從昨日有整理此文的動機後,到今日上午找電腦上網而不得,再到此刻在網吧完成此文。着實也體味了一把什麼叫作爲技術狂熱的感受。大型 網站架構是一個實戰性很強的東西,而你我或許如今暫時還只是一個在外看熱鬧的門外漢而已。不過,不要緊,小魚小蝦照樣能暢遊汪汪大洋,更況且往後亦能成長 爲大魚大鯊。