為什麼說資料庫連接很消耗資源

来源:https://www.cnblogs.com/tyson03/archive/2023/03/30/17274731.html
-Advertisement-
Play Games

相信有過工作經驗的同學都知道資料庫連接是一個比較耗資源的操作。那麼資源到底是耗費在哪裡呢? 本文主要想探究一下連接資料庫的細節,尤其是在Web應用中要使用資料庫來連接池,以免每次發送一次請求就重新建立一次連接。對於這個問題,答案都是一致的,建立資料庫連接很耗時,但是這個耗時是都多少呢,又是分別在哪些 ...


相信有過工作經驗的同學都知道資料庫連接是一個比較耗資源的操作。那麼資源到底是耗費在哪裡呢?

本文主要想探究一下連接資料庫的細節,尤其是在Web應用中要使用資料庫來連接池,以免每次發送一次請求就重新建立一次連接。對於這個問題,答案都是一致的,建立資料庫連接很耗時,但是這個耗時是都多少呢,又是分別在哪些方面產生的耗時呢?

本文以連接MySQL資料庫為例,因為MySQL資料庫是開源的,其通信協議是公開的,所以我們能夠詳細分析建立連接的整個過程。

在本文中,消耗資源的分析主要集中在網路上,當然,資源也包括記憶體、CPU等計算資源,使用的編程語言是Java,但是不排除編程語言也會有一定的影響。

首先先看一下連接資料庫的Java代碼,如下:

Class.forName("com.mysql.jdbc.Driver");

String name = "xttblog2";
String password = "123456";
String url = "jdbc:mysql://xxx:3306/xttblog2";
Connection conn = DriverManager.getConnection(url, name, password);
// 之後程式終止,連接被強制關閉

然後通過Wireshark,分析整個連接的建立過程,如下:

本文已經收錄到Github倉庫,該倉庫包含電腦基礎、Java基礎、多線程、JVM、資料庫、Redis、Spring、Mybatis、SpringMVC、SpringBoot、分散式、微服務、設計模式、架構、校招社招分享等核心知識點,歡迎star~

Github地址

如果訪問不了Github,可以訪問gitee地址。

gitee地址

Wireshark抓包

在上圖中顯示的連接過程中,可以看出MySQL的通信協議是基於TCP傳輸協議的,而且該協議是二進位協議,不是類似於HTTP的文本協議,其中建立連接的過程具體如下:

  • 第1步:建立TCP連接,通過三次握手實現;
  • 第2步:伺服器發送給客戶端「握手信息」 ,客戶端響應該握手消息;
  • 第3步:客戶端「發送認證包」 ,用於用戶驗證,驗證成功後,伺服器返回OK響應,之後開始執行命令;

用戶驗證成功之後,會進行一些連接變數的設置,比如字元集、是否自動提交事務等,其間會有多次數據的交互。完成了這些步驟後,才會執行真正的數據查詢和更新等操作。

在本文的測試中,只用了5行代碼來建立連接,但是並沒有通過該連接去執行任何操作,所以在程式執行完畢之後,連接不是通過Connection.close()關閉的,而是由於程式執行完畢,導致進程終止,造成與資料庫的連接異常關閉,所以最後會出現TCPRST報文。

在這個最簡單的代碼中,沒有設置任何額外的連接屬性,所以在設置屬性上占用的時間可以認為是最少的(其實,雖然我們沒有設置任何屬性,但是驅動仍然設置了字元集、事務自動提交等,這取決於具體的驅動實現),所以整個連接所使用的時間可以認為是最少的。

但從統計信息中可以看出,在不包括最後TCPRST報文時(因為該報文不需要伺服器返回任何響應),但是其中仍需在客戶端和伺服器之間進行往返「7」 次,「也就是說完成一次連接,可以認為,數據在客戶端和伺服器之間需要至少往返7次」 ,從時間上來看,從開始TCP的三次握手,到最終連接強制斷開為止(不包括最後的RST報文),總共花費了:

10.416042 - 10.190799 = 0.225243s = 225.243ms!!!

這意味著,建立一次資料庫連接需要225ms,而這還是還可以認為是最少的,當然「花費的時間可能受到網路狀況、資料庫伺服器性能以及應用代碼是否高效的影響」 ,但是這裡只是一個最簡單的例子,已經足夠說明問題了!

由於上面是程式異常終止了,但是在正常的應用程式中,連接的關閉一般都是通過Connection.close()完成的,代碼如下:

Class.forName("com.mysql.jdbc.Driver");

String name = "shine_user";
String password = "123";
String url = "jdbc:mysql://xxx:3306/clever_mg_test";
Connection conn = DriverManager.getConnection(url, name, password);
conn.close();

這樣的話,情況發生了變化,主要體現在與資料庫連接的斷開,如下圖:

網路抓包

  • 第1步:此時處於MySQL通信協議階段,客戶端發送關閉連接請求,而且不用等待服務端的響應;
  • 第2步:TCP斷開連接,4次揮手完成連接斷開;

這裡是完整地完成了從資料庫連接的建立到關閉,整個過程花費了:

747.284311 - 747.100954 = 0.183357s = 183.357ms

這裡可能也有網路狀況的影響,比上述的225ms少了,但是也幾乎達到了200ms的級別。最全面的Java面試網站

那麼問題來了,想象一下這個場景,對於一個日活2萬的網站來說,假設每個用戶只會發送5個請求,那麼一天就是10萬個請求,對於建立資料庫連接,我們保守一點計算為150ms好了,那麼一天當中花費在建立資料庫連接的時間有(還不包括執行查詢和更新操作):

100000 * 150ms = 15000000ms = 15000s = 250min = 4.17h

也就說每天花費在建立資料庫連接上的時間已經達到「4個小時」 ,所以說資料庫連接池是必須的,而且當日活增加時,單單使用資料庫連接池也不能完全保證你的服務能夠正常運行,還需要考慮其他的解決方案:

  • 緩存
  • SQL 的預編譯
  • 負載均衡
  • ……

總之,資料庫連接真的很耗時,所以不要頻繁的建立連接

最後給大家分享一個Github倉庫,上面有大彬整理的300多本經典的電腦書籍PDF,包括C語言、C++、Java、Python、前端、資料庫、操作系統、電腦網路、數據結構和演算法、機器學習、編程人生等,可以star一下,下次找書直接在上面搜索,倉庫持續更新中~

Github地址


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 這是一款使用 ChatGPT API 進行劃詞翻譯和文本潤色的瀏覽器插件。藉助了 ChatGPT 強大的翻譯能力,它將幫助您更流暢地閱讀外語和編輯外語。 它能幹啥 一. 可翻譯 二. 可潤色 三. 可總結 四. 可分析 五. 可解釋程式代碼 插件地址 OpenAI Translator 勸退聲明 由 ...
  • 一句話來解釋什麼是深淺拷貝,B拷貝A,當修改A,B如果變化,就是淺拷貝,反之就是深拷貝。 基本原理: 1.遞歸函數2.對象內的值都是簡單數據類型時 直接進行賦值3.當我們遇到數組和對象時,可以再次調用函數,利用遞歸去拷貝數組和對象內的每個值4.先數組 後對象 因為數組也是對象 下麵是一個實現深拷貝的 ...
  • 我們是袋鼠雲數棧 UED 團隊,致力於打造優秀的一站式數據中台產品。我們始終保持工匠精神,探索前端道路,為社區積累並傳播經驗價值。 本文作者:景明 我們以一段 C 代碼為例,來看一下代碼被編譯成二進位可執行程式之後,是如何被 CPU 執行的。 在這段代碼中,只是做了非常簡單的加法操作,將 x 和 y ...
  • 前言 前面我們簡單的瞭解了 vue 初始化時的一些大概的流程,這裡我們詳細的瞭解下具體的內容; 內容 這一塊主要圍繞init.ts中的vm.$mount進行剖析。 vm.$mount vm.$mount是全局的公共方法方法,但是這是我們要找的話就要向上查找了,代碼位於scr/platforms/we ...
  • 依賴註入模式(Dependency Injection Pattern):允許我們通過將對象的依賴關係從代碼中分離出來,從而使代碼更加模塊化和可重用。 在傳統的編程模式中,一個對象可能會直接創建或者獲取它需要的其他對象,這樣會造成對象之間的緊耦合關係,難以維護和擴展。而使用依賴註入模式,則可以將對象 ...
  • 簡介 命令模式(Command Pattern)是一種數據驅動的設計模式,也是一種行為型設計模式。這種模式的請求以命令的形式包裹在對象中,並傳給調用對象。調用對象再尋找合適的對象,並把該命令傳給相應的處理者。即把請求或操作封裝成單個對象,並使其可以被參數化和延遲執行,這種方式將命令和執行者進行了有效 ...
  • 經常會聽到開發者提起單元測試的話題,那麼今天我就帶大伙一起來看看大名鼎鼎的谷歌 C++ 測試框架 GoogleTest. ...
  • 緒論 本文將介紹一個完全用Verilog HDL手寫的AMBA片上系統,項目的主題是設計一個基於AMBA匯流排的流水燈控制系統, 項目中所有數字電路邏輯都將通過Verilog進行RTL設計,不會調用成熟IP核, 然後利用Vivado平臺對RTL模型進行模擬、綜合與佈線,最後在FPGA開發板上進行板級驗 ...
一周排行
    -Advertisement-
    Play Games
  • ## 引言 最近發現自己喜歡用的 Todo 軟體總是差點意思,畢竟每個人的習慣和工作流不太一樣,我就想著自己寫一個小的[Todo 項目]( https://github.com/circler3/TodoTrack ),核心的功能是自動記錄 Todo 執行過程中消耗的時間(尤其面向程式員),按照自己 ...
  • ### 前言 當我們編寫 C# 代碼時,經常需要處理大量的數據集合。在傳統的方式中,我們往往需要先將整個數據集合載入到記憶體中,然後再進行操作。但是如果數據集合非常大,這種方式就會導致記憶體占用過高,甚至可能導致程式崩潰。 C# 中的`yield return`機制可以幫助我們解決這個問題。通過使用`y ...
  • 1. ADO.NET的前世今生 ADO.NET的名稱起源於ADO(ActiveX Data Objects),是一個COM組件庫,用於在以往的Microsoft技術中訪問數據。之所以使用ADO.NET名稱,是因為Microsoft希望表明,這是在NET編程環境中優先使用的數據訪問介面。 ADO.NE ...
  • 1. 為什麼需要單元測試 在我們之前,測試某些功能是否能夠正常運行時,我們都將代碼寫到Main方法中,當我們測試第二個功能時,我們只能選擇將之前的代碼清掉,重新編寫。此時,如果你還想重新測試你之前的功能時,這時你就顯得有些難為情了,因為代碼都被你清掉了。當然你完全可以把代碼寫到一個記事本中進行記錄, ...
  • 1. 透過現象看本質 反射被譽為是 c#中的黑科技 ,在很多領域中都有反射的身影,例如,我們經常使用的ORM框架,ABP框架 等。 反射指程式可以訪問、檢測和修改它本身狀態或行為的一種能力。. 程式集包含模塊,而模塊包含類型,類型又包含成員。. 反射則提供了封裝程式集、模塊和類型的對象。. 您可以使 ...
  • # Rust Web 全棧開發之 Web Service 中的錯誤處理 ## Web Service 中的統一錯誤處理 ### Actix Web Service 自定義錯誤類型 -> 自定義錯誤轉為 HTTP Response - 資料庫 - 資料庫錯誤 - 串列化 - serde 錯誤 - I/ ...
  • 在前面的幾篇文章中,詳細地給大家介紹了Java里的集合。但在介紹集合時,我們涉及到了泛型的概念卻並沒有詳細學習,所以今天我們要花點時間給大家專門講解什麼是泛型、泛型的作用、用法、特點等內容 ...
  • ###BIO:同步阻塞 主線程發起io請求後,需要等待當前io操作完成,才能繼續執行。 ###NIO:同步非阻塞 引入selector、channel、等概念,當主線程發起io請求後,輪詢的查看系統是否準備好執行io操作,沒有準備好則主線程不會阻塞會繼續執行,準備好主線程會阻塞等待io操作完成。 # ...
  • 摘要:在讀多寫少的環境中,有沒有一種比ReadWriteLock更快的鎖呢?有,那就是JDK1.8中新增的StampedLock! 本文分享自華為雲社區《【高併發】高併發場景下一種比讀寫鎖更快的鎖》,作者: 冰 河。 什麼是StampedLock? ReadWriteLock鎖允許多個線程同時讀取共 ...
  • ## 併發與並行😣 ### 併發與並行的概念和區別 並行:同一個時間段內多個任務同時在不同的CPU核心上執行。強調同一時刻多個任務之間的”**同時執行**“。 併發:同一個時間段內多個任務都在進展。強調多個任務間的”**交替執行**“。 ![](https://img2023.cnblogs.co ...