?一、網站應用背景

開發一個網站的應用程序,當用戶規模比較小的時候,使用簡單的:一台應用伺服器+一台資料庫伺服器+一台文件伺服器,這樣的話完全可以解決一部分問題,也可以通過堆硬體的方式來提高網站應用的訪問性能,當然,也要考慮成本的問題。

當問題的規模在經濟條件下通過堆硬體的方式解決不了的時候,我們應該通過其他的思路去解決問題,互聯網發展至今,已經提供了很多成熟的解決方案,但並不是都具有適用性,你把淘寶的技術全部都搬過來也不一定達到現在淘寶的水平,道理很簡單。

當然,很多文章都在強調,一個網站的發展水平,是逐漸的演變過來的,並不是一朝一夕的事情。雖然目前的情況互聯網的泡沫越來越大,但是整個互聯網技術的發展確實為我們提供了方便快捷的上網體驗。下邊是一張早期的淘寶官網的界面:

目前,博主正在跟隨導師做一個創業項目,使用的技術是SSM+MySQL+Linux這些,但是由於資金的限制和考慮到用戶群體的特殊性,系統的架構無奈的選擇的就是最簡單的方式:一台應用伺服器、一台資料庫伺服器、一台文件系統伺服器,沒有用到高級的技術,也沒有用到分散式部署的方案。下邊整理的是一些針對海量數據和高並發情況下的解決方案,技術水平有限,歡迎留言指導。

二、針對海量數據和高並發的主要解決方案

2.1、海量數據的解決方案:

  1. 使用緩存;
  2. 頁面靜態化技術;
  3. 資料庫優化;
  4. 分離資料庫中活躍的數據;
  5. 批量讀取和延遲修改;
  6. 讀寫分離;
  7. 使用NoSQL和Hadoop等技術;
  8. 分散式部署資料庫;
  9. 應用服務和數據服務分離;
  10. 使用搜索引擎搜索資料庫中的數據;
  11. 進行業務的拆分;

2.2、高並發情況下的解決方案:

  1. 應用程序和靜態資源文件進行分離;
  2. 頁面緩存;
  3. 集群與分散式;
  4. 反向代理;
  5. CDN;

三、海量數據的解決方案

3.1、使用緩存

網站訪問數據的特點大多數呈現為「二八定律」:80%的業務訪問集中在20%的數據上。

例如:在某一段時間內百度的搜索熱詞可能集中在少部分的熱門辭彙上;新浪微博某一時期也可能大家廣泛關注的主題也是少部分事件。

總的來說就是用戶只用到了總數據條目的一小部分,當網站發展到一定規模,資料庫IO操作成為性能瓶頸的時候,使用緩存將這一小部分的熱門數據緩存在內存中是一個很不錯的選擇,不但可以減輕資料庫的壓力,還可以提高整體網站的數據訪問速度。

使用緩存的方式可以通過程序代碼將數據直接保存到內存中,例如通過使用Map或者ConcurrentHashMap;另一種,就是使用緩存框架:Redis、Ehcache、Memcache等。

使用緩存框架的時候,我們需要關心的就是什麼時候創建緩存緩存失效策略

緩存的創建可以通過很多方式進行創建,具體也需要根據自己的業務進行選擇。例如,新聞首頁的新聞應該在第一次讀取數據的時候就進行緩存;對於點擊率比較高的文章,可以將其文章內容進行緩存等。

內存資源有限,選擇如何創建緩存是一個值得思考的問題。另外,對於緩存的失效機制也是需要好好研究的,可以通過設置失效時間的方式進行設置;也可以通過對熱門數據設置優先順序,根據不同的優先順序設置不同的失效時間等;

需要注意的是,當我們刪除一條數據的時候,我們要考慮到刪除該條緩存,還要考慮在刪除該條緩存之前該條數據是否已經到達緩存失效時間等各種情況!

使用緩存的時候還要考慮到緩存伺服器發生故障時候如何進行容錯處理,是使用N多台伺服器緩存相同的數據,通過分散式部署的方式對緩存數據進行控制,當一台發生故障的時候自動切換到其他的機器上去;還是通過Hash一致性的方式,等待緩存伺服器恢復正常使用的時候重新指定到該緩存伺服器。Hash一致性的另一個作用就是在分散式緩存伺服器下對數據進行定位,將數據分布在不用緩存伺服器上。關於數據緩存的Hash一致性也是一個比較打的問題,這裡只能大致描述一下,關於Hash一致性的了解,推薦一篇文章:blog.csdn.net/liu765023

3.2、頁面靜態化技術

使用傳統的JSP界面,前端界面的顯示是通過後台伺服器進行渲染後返回給前端遊覽器進行解析執行,如下圖:

當然,現在提倡前後端分離,前端界面基本都是HTML網頁代碼,通過Angular JS或者NodeJS提供的路由向後端伺服器發出請求獲取數據,然後在遊覽器對數據進行渲染,這樣在很大程度上降低了後端伺服器的壓力。

還可以將這些靜態的HTML、CSS、JS、圖片資源等放置在緩存伺服器上或者CDN伺服器上,一般使用最多的應該是CDN伺服器或者Nginx伺服器提供的靜態資源功能。

另外,在《高性能網站建設進階指南-Web開發者性能優化最佳實踐(口碑網前端團隊 翻譯)》這本書中,對網站性能的前端界面提供了一些很寶貴的經驗,如下:

因此,在這些靜態資源的處理上,選擇正確的處理方式還是對整體網站性能還是有很大幫助的!

3.3、資料庫優化

資料庫優化是整個網站性能優化的最基礎的一個環節,因為,大多數網站性能的瓶頸都是開在資料庫IO操作上,雖然提供了緩存技術,但是對資料庫的優化還是一個需要認真的對待。一般公司都有自己的DBA團隊,負責資料庫的創建,數據模型的確立等問題,不像我們現在幾個不懂資料庫優化的人只能在網上找一篇篇資料庫優化的文章,自己去摸索,並沒有形成一個系統的資料庫優化思路。

對於資料庫的優化來說,是一種用技術換金錢的方式。資料庫優化的方式很多,常見的可以分為:資料庫表結構優化、SQL語句優化、分區、分表、索引優化、使用存儲過程代替直接操作等 。

3.3.1、表結構優化

對於資料庫的 開發規範與使用技巧以及設計和優化,前邊的時候總結了一些文章,這裡偷個懶直接放地址,有需要的可以移步看一下:

a) MySQL開發規範與使用技巧總結:http://blog.csdn.net/xlgen157387/article/details/48086607

b) 在一個千萬級的資料庫查尋中,如何提高查詢效率?:http://blog.csdn.net/xlgen157387/article/details/44156679

另外,再設計資料庫表的時候需不需要創建外鍵,使用外鍵的好處之一可以方便的進行級聯刪除操作,但是現在在進行數據業務操作的時候,我們都通過事物的方式來保證數據讀取操作的一致性,我感覺相比於使用外鍵關聯MySQL自動幫我們完成級聯刪除的操作來說,還是自己使用事物進行刪除操作來的更放心一些。當然可能也是有適用的場景,大家如有很好的建議,歡迎留言!

3.3.2、SQL優化

對於SQL的優化,主要是針對SQL語句處理邏輯的優化,而且還要根據索引進行配合使用。另外,對於SQL語句的優化我們可以針對具體的業務方法進行優化,我們可以將執行業務邏輯操作的資料庫執行時間記錄下來,來進行有針對性的優化,這樣的話效果還是很不錯的!例如下圖,展示了一條資料庫操作執行調用的時間:

關於SQL優化的一些建議,以前整理了一些,還請移步查看:

a) 19個MySQL性能優化要點解析:http://blog.csdn.net/xlgen157387/article/details/50735269

b) MySQL批量SQL插入各種性能優化:http://blog.csdn.net/xlgen157387/article/details/50949930

3.3.3、分表

分表是將一個大表按照一定的規則分解成多張具有獨立存儲空間的實體表,我們可以稱為子表,每個表都對應三個文件,MYD數據文件,.MYI索引文件,.frm表結構文件。這些子表可以分布在同一塊磁碟上,也可以在不同的機器上。資料庫讀寫操作的時候根據事先定義好的規則得到對應的子表名,然後去操作它。

例如:用戶表 用戶的角色有很多種,可以通過枚舉類型的方式將用戶分為不同類別category:學生、教師、企業等 ,這樣的話,我們就可以根據類別category來對資料庫進行分表,這樣的話每次查詢的時候現根據用戶的類型鎖定一個較小的範圍。

不過分表之後,如果需要查詢完整的順序就需要使用多表操作了。

3.3.4、分區

資料庫分區是一種物理資料庫設計技術,DBA和資料庫建模人員對其相當熟悉。雖然分區技術可以實現很多效果,但其主要目的是為了在特定的SQL操作中減少數據讀寫的總量以縮減響應時間。

分區和分表相似,都是按照規則分解表。不同在於分表將大表分解為若干個獨立的實體表,而分區是將數據分段劃分在多個位置存放,可以是同一塊磁碟也可以在不同的機器。分區後,表面上還是一張表,但數據散列到多個位置了。資料庫讀寫操作的時候操作的還是大表名字,DMS自動去組織分區的數據。

當一張表中的數據變得很大的時候,讀取數據,查詢數據的效率非常低下,很容易的就是講數據分到不同的數據表中進行保存,但是這樣分表之後會使得操作起來比較麻煩,因為,將同類的數據分別放在不同的表中的話,在搜索數據的時候需要便利查詢這些表中的數據。想進行CRUD操作還需要先找到對應的所有表,如果涉及到不同的表的話還要進行跨表操作,這樣操作起來還是很麻煩的。

使用分區的方式可以解決這個問題,分區是將一張表中的數據按照一定的規則分到不同的區中進行保存,這樣進行數據查詢的時候如果數據的範圍在同一個區域內那麼就可以支隊一個區中的數據進行操作,這樣的話操作起來數據量更少,操作速度更快,而且該方法是對程序透明的,程序不需要進行任何的修改。

3.3.5、索引優化

索引的大致原理是在數據發生變化的時候就預先按指定欄位的順序排列後保存到一個類似表的結構中,這樣在查找索引欄位為條件記錄時就可以很快地從索引中找到對應記錄的指針並從表中獲取到相應的數據,這樣速度是很快地。

不過,雖然查詢的效率大大提高了,但是在進行增刪改的時候,因為數據的變化都需要更新相應的索引,也是一種資源的浪費。

關於使用索引的問題,對待不同的問題,還是需要進行不同的討論,根據具體的業務需求選擇合適的索引對性能的提高效果是很明顯的一個舉措!

推薦文章閱讀:

a) 資料庫索引的作用和優點缺點以及索引的11中用法:http://blog.csdn.net/xlgen157387/article/details/45030829

b) 資料庫索引原理:http://blog.csdn.net/kennyrose/article/details/7532032

3.3.6、使用存儲過程代替直接操作

存儲過程(Stored Procedure)是在大型資料庫系統中,一組為了完成特定功能的SQL 語句集,存儲在資料庫中,經過第一次編譯後再次調用不需要再次編譯,用戶通過指定存儲過程的名字並給出參數(如果該存儲過程帶有參數)來執行它。存儲過程是資料庫中的一個重要對象,任何一個設計良好的資料庫應用程序都應該用到存儲過程。

在操作過程比較複雜並且調用頻率比較高的業務中,可以將編寫好的sql語句用存儲過程的方式來代替,使用存儲過程只需要進行一次變異,而且可以在一個存儲過程里做一些複雜的操作。

3.4、分離資料庫中活躍的數據

正如前邊提到的「二八定律」一樣,網站的數據雖然很多,但是經常被訪問的數據還是有限的,因此可以講這些相對活躍的數據進行分離出來單獨進行保存來提高處理效率。

其實前邊使用緩存的思想就是一個很明顯的分離資料庫中活躍的數據的使用案例,將熱門數據緩存在內存中。

還有一種場景就是,例如一個網站的所用註冊用戶量很大千萬級別,但是經常登錄的用戶只有百萬級別,剩下的基本都是很長時間都沒有進行登錄操作,如果不把這些「殭屍用戶」單獨分離出去,那麼我們每次查詢其他登錄用戶的時候,就白白浪費了這些殭屍用戶的查詢操作。

3.5、批量讀取和延遲修改

批量讀取和延遲修改的原理是通過減少操作資料庫的操作來提高效率。

批量讀取是將多次查詢合併到一次中進行讀取,因為每一個資料庫的請求操作都需要鏈接的建立和鏈接的釋放,還是佔用一部分資源的,批量讀取可以通過非同步的方式進行讀取。

延遲修改是對於一些高並發的並且修改頻繁修改的數據,在每次修改的時候首先將數據保存到緩存中,然後定時將緩存中的數據保存到資料庫中,程序可以在讀取數據時可以同時讀取資料庫中和緩存中的數據。

例如:我現在在編寫這份博客,我一開始寫了一寫內容然後點擊發布了,在次回到Markdown進行修改操作,我有一個習慣,沒寫一些東西總是按一下上邊的 「保存」按鈕,然後我在另一個頁面打開這篇博客查看,我的修改已經被更新,但是我還在 編輯!

不知道CSDN的技術是不是在我沒有點擊發布之前,這些數據都是先放到緩存裡邊的。

3.6、讀寫分離

讀寫分離的實質是將應用程序對資料庫的讀寫操作分配到多個資料庫伺服器上,從而降低單台資料庫的訪問壓力。

讀寫分離一般通過配置主從資料庫的方式,數據的讀取來自從庫,對資料庫增加修改刪除操作主庫。

相關文章請移步觀看:

a) MySQL5.6 資料庫主從(Master/Slave)同步安裝與配置詳解:http://blog.csdn.net/xlgen157387/article/details/51331244

b) MySQL主從複製的常見拓撲、原理分析以及如何提高主從複製的效率總結:http://blog.csdn.net/xlgen157387/article/details/52451613

3.7、使用NoSQL和Hadoop等技術

NoSQL是一種非結構化的非關係型資料庫,由於其靈活性,突破了關係型資料庫的條條框框,可以靈活的進行操作,另外,因為NoSQL通過多個塊存儲數據的特點,其操作大數據的速度也是相當快的。

3.8、分散式部署資料庫

任何強大的單一伺服器都滿足不了大型網站持續增長的業務需求。資料庫通過讀寫分離之後將一台資料庫伺服器拆分為兩台或者多台資料庫伺服器,但是仍然滿足不了持續增長的業務需求。分散式部署資料庫是將網站資料庫拆分的最後手段,只有在單表數據規模非常龐大的時候才使用。

分散式部署資料庫是一種很理想的情況,分散式資料庫是將表存放在不同的資料庫中然後再放到不同的資料庫中,這樣在處理請求的時候,如果需要調用多個表,則可以讓多台伺服器同時處理,從而提高處理效率。

分散式資料庫簡單的架構圖如下:

3.9、應用服務和數據服務分離

應用伺服器和資料庫伺服器進行分離的目的是為了根據應用伺服器的特點和資料庫伺服器的特點進行底層的優化,這樣的話能夠更好的發揮每一台伺服器的特性,資料庫伺服器當然是有一定的磁碟空間,而應用伺服器相對不需要太大的磁碟空間,這樣的話進行分離是有好處的,也能防止一台伺服器出現問題連帶的其他服務也不可以使用。

3.10、使用搜索引擎搜索資料庫中的數據

使用搜索引擎這種非資料庫查詢技術對網站應用的可伸縮分散式特性具有更好的支持。

常見的搜索引擎如Solr通過一種反向索引的方式,維護關鍵字到文檔的映射關係,類似於我們使用《新華字典》進行搜索一個關鍵字,首先應該是看字典的目錄進行查找然後定位到具體的位置。

搜索引擎通過維護一定的關鍵字到文檔的映射關係,能夠快速的定位到需要查找的數據,相比於傳統的資料庫搜索的方式,效率還是很高的。

目前一種比較火的ELK stack技術,還是值得學習的。

一篇關於Solr與MySQL查詢性能對比文章: Solr與MySQL查詢性能對比:http://www.cnblogs.com/luxiaoxun/p/4696477.html?utm_source=tuicool&utm_medium=referral

3.11、進行業務的拆分

為什麼進行業務的拆分,歸根結底上還是使用的還是講不通的業務數據表部署到不用的伺服器上,分別查找對應的數據以滿足網站的需求。各個應用之間用過指定的URL連接獲取不同的服務,

例如一個大型的購物網站就會將首頁、商鋪、訂單、買家、賣家等拆分為不通的子業務,一方面將業務模塊分歸為不同的團隊進行開發,另外一方面不同的業務使用的資料庫表部署到不通的伺服器上,體現到拆分的思想,當一個業務模塊使用的資料庫伺服器發生故障也不會影響其他業務模塊的資料庫正常使用。另外,當其中一個模塊的訪問量激增的時候還可以動態的擴展這個模塊使用到的資料庫的數量從而滿足業務的需求。

四、高並發情況下的解決方案

4.1、應用程序和靜態資源文件進行分離

所謂的靜態資源就是我們網站中用到的Html、Css、Js、Image、Video、Gif等靜態資源。應用程序和靜態資源文件進行分離也是常見的前後端分離的解決方案,應用服務只提供相應的數據服務,靜態資源部署在指定的伺服器上(Nginx伺服器或者是CDN伺服器上),前端界面通過Angular JS或者Node JS提供的路由技術訪問應用伺服器的具體服務獲取相應的數據在前端遊覽器上進行渲染。這樣可以在很大程度上減輕後端伺服器的壓力。

例如,百度主頁使用的圖片就是單獨的一個域名伺服器上進行部署的

4.2、頁面緩存

頁面緩存是將應用生成的很少發生數據變化的頁面緩存起來,這樣就不需要每次都重新生成頁面了,從而節省大量CPU資源,如果將緩存的頁面放到內存中速度就更快。

可以使用Nginx提供的緩存功能,或者可以使用專門的頁面緩存伺服器Squid。

4.3、集群與分散式

4.4、反向代理

參考文章請移步至: blog.csdn.net/xlgen1573

4.5、CDN

CDN伺服器其實是一種集群頁面緩存伺服器,其目的就是儘早的返回用戶所需要的數據,一方面加速用戶訪問速度,另一方面也減輕後端伺服器的負載壓力。

CDN的全稱是Content Delivery Network,即內容分發網路。其基本思路是儘可能避開互聯網上有可能影響數據傳輸速度和穩定性的瓶頸和環節,使內容傳輸的更快、更穩定。

CDN通過在網路各處放置節點伺服器所構成的在現有的互聯網基礎之上的一層智能虛擬網路,CDN系統能夠實時地根據網路流量和各節點的連接、負載狀況以及到用戶的距離和響應時間等綜合信息將用戶的請求重新導向離用戶最近的服務節點上。其目的是使用戶可就近取得所需內容,解決 Internet網路擁擠的狀況,提高用戶訪問網站的響應速度。

也就是說CDN伺服器是部署在網路運行商的機房,提供的離用戶最近的一層數據訪問服務,用戶在請求網站服務的時候,可以從距離用戶最近的網路提供商機房獲取數據。電信的用戶會分配電信的節點,聯通的會分配聯通的節點。

CDN分配請求的方式是特殊的,不是普通的負載均衡伺服器來分配的那種,而是用專門的CDN域名解析伺服器在解析與名的時候就分配好的。

CDN結構圖如下所示:

五、總結

文章提到的幾點並沒有詳細的介紹,如需要對其中的一種方式進行研究,可以自行去找資源學習研究,這裡只起到拋磚引玉的作用。當然對於大型網站應用之海量數據和高並發解決方案不局限於這些技巧或技術,還有很多成熟的解決方案,有需要的可以自行了解。

特此說明:本文的配圖來自《網站架構及其演變過程--韓路彪》,多謝原作者提供的精美配圖,並且文章的結構大致也參考了作者的思路,只不過內容是自己的一些經驗和學習到的東西進行整理的。

參考書籍或文章:

1、《網站架構及其演變過程--韓路彪》

2、《大型網站技術架構之核心原理與參考案例--李智慧》

3、部分專業名詞簡介來自百度百科

4、cio.chinabyte.com/428/1

5、blog.codinglabs.org/art

還有一些在整理的時候,忘記記錄連接,還請大家見諒!


搜索或掃描下述二維碼關注微信公眾號:Java後端技術(ID: JavaITWork),和20萬人一起學Java!

Java後端技術專註Java相關技術:SSM、Spring全家桶、微服務、MySQL、MyCat、集群、分散式、中間件、Linux、網路、多線程,偶爾講點運維Jenkins、Nexus、Docker、ELK,偶爾分享些技術乾貨,致力於Java全棧開發!


推薦閱讀:
相关文章