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

来源: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
  • 概述:本文代碼示例演示瞭如何在WPF中使用LiveCharts庫創建動態條形圖。通過創建數據模型、ViewModel和在XAML中使用`CartesianChart`控制項,你可以輕鬆實現圖表的數據綁定和動態更新。我將通過清晰的步驟指南包括詳細的中文註釋,幫助你快速理解並應用這一功能。 先上效果: 在 ...
  • openGauss(GaussDB ) openGauss是一款全面友好開放,攜手伙伴共同打造的企業級開源關係型資料庫。openGauss採用木蘭寬鬆許可證v2發行,提供面向多核架構的極致性能、全鏈路的業務、數據安全、基於AI的調優和高效運維的能力。openGauss深度融合華為在資料庫領域多年的研 ...
  • openGauss(GaussDB ) openGauss是一款全面友好開放,攜手伙伴共同打造的企業級開源關係型資料庫。openGauss採用木蘭寬鬆許可證v2發行,提供面向多核架構的極致性能、全鏈路的業務、數據安全、基於AI的調優和高效運維的能力。openGauss深度融合華為在資料庫領域多年的研 ...
  • 概述:本示例演示了在WPF應用程式中實現多語言支持的詳細步驟。通過資源字典和數據綁定,以及使用語言管理器類,應用程式能夠在運行時動態切換語言。這種方法使得多語言支持更加靈活,便於維護,同時提供清晰的代碼結構。 在WPF中實現多語言的一種常見方法是使用資源字典和數據綁定。以下是一個詳細的步驟和示例源代 ...
  • 描述(做一個簡單的記錄): 事件(event)的本質是一個委托;(聲明一個事件: public event TestDelegate eventTest;) 委托(delegate)可以理解為一個符合某種簽名的方法類型;比如:TestDelegate委托的返回數據類型為string,參數為 int和 ...
  • 1、AOT適合場景 Aot適合工具類型的項目使用,優點禁止反編 ,第一次啟動快,業務型項目或者反射多的項目不適合用AOT AOT更新記錄: 實實在在經過實踐的AOT ORM 5.1.4.117 +支持AOT 5.1.4.123 +支持CodeFirst和非同步方法 5.1.4.129-preview1 ...
  • 總說周知,UWP 是運行在沙盒裡面的,所有許可權都有嚴格限制,和沙盒外交互也需要特殊的通道,所以從根本杜絕了 UWP 毒瘤的存在。但是實際上 UWP 只是一個應用模型,本身是沒有什麼許可權管理的,許可權管理全靠 App Container 沙盒控制,如果我們脫離了這個沙盒,UWP 就會放飛自我了。那麼有沒... ...
  • 目錄條款17:讓介面容易被正確使用,不易被誤用(Make interfaces easy to use correctly and hard to use incorrectly)限制類型和值規定能做和不能做的事提供行為一致的介面條款19:設計class猶如設計type(Treat class de ...
  • title: 從零開始:Django項目的創建與配置指南 date: 2024/5/2 18:29:33 updated: 2024/5/2 18:29:33 categories: 後端開發 tags: Django WebDev Python ORM Security Deployment Op ...
  • 1、BOM對象 BOM:Broswer object model,即瀏覽器提供我們開發者在javascript用於操作瀏覽器的對象。 1.1、window對象 視窗方法 // BOM Browser object model 瀏覽器對象模型 // js中最大的一個對象.整個瀏覽器視窗出現的所有東西都 ...