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

来源: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
  • 前言 在我們開發過程中基本上不可或缺的用到一些敏感機密數據,比如SQL伺服器的連接串或者是OAuth2的Secret等,這些敏感數據在代碼中是不太安全的,我們不應該在源代碼中存儲密碼和其他的敏感數據,一種推薦的方式是通過Asp.Net Core的機密管理器。 機密管理器 在 ASP.NET Core ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 順序棧的介面程式 目錄順序棧的介面程式頭文件創建順序棧入棧出棧利用棧將10進位轉16進位數驗證 頭文件 #include <stdio.h> #include <stdbool.h> #include <stdlib.h> 創建順序棧 // 指的是順序棧中的元素的數據類型,用戶可以根據需要進行修改 ...
  • 前言 整理這個官方翻譯的系列,原因是網上大部分的 tomcat 版本比較舊,此版本為 v11 最新的版本。 開源項目 從零手寫實現 tomcat minicat 別稱【嗅虎】心有猛虎,輕嗅薔薇。 系列文章 web server apache tomcat11-01-官方文檔入門介紹 web serv ...
  • C總結與剖析:關鍵字篇 -- <<C語言深度解剖>> 目錄C總結與剖析:關鍵字篇 -- <<C語言深度解剖>>程式的本質:二進位文件變數1.變數:記憶體上的某個位置開闢的空間2.變數的初始化3.為什麼要有變數4.局部變數與全局變數5.變數的大小由類型決定6.任何一個變數,記憶體賦值都是從低地址開始往高地 ...
  • 如果讓你來做一個有狀態流式應用的故障恢復,你會如何來做呢? 單機和多機會遇到什麼不同的問題? Flink Checkpoint 是做什麼用的?原理是什麼? ...
  • C++ 多級繼承 多級繼承是一種面向對象編程(OOP)特性,允許一個類從多個基類繼承屬性和方法。它使代碼更易於組織和維護,並促進代碼重用。 多級繼承的語法 在 C++ 中,使用 : 符號來指定繼承關係。多級繼承的語法如下: class DerivedClass : public BaseClass1 ...
  • 前言 什麼是SpringCloud? Spring Cloud 是一系列框架的有序集合,它利用 Spring Boot 的開發便利性簡化了分散式系統的開發,比如服務註冊、服務發現、網關、路由、鏈路追蹤等。Spring Cloud 並不是重覆造輪子,而是將市面上開發得比較好的模塊集成進去,進行封裝,從 ...
  • class_template 類模板和函數模板的定義和使用類似,我們已經進行了介紹。有時,有兩個或多個類,其功能是相同的,僅僅是數據類型不同。類模板用於實現類所需數據的類型參數化 template<class NameType, class AgeType> class Person { publi ...
  • 目錄system v IPC簡介共用記憶體需要用到的函數介面shmget函數--獲取對象IDshmat函數--獲得映射空間shmctl函數--釋放資源共用記憶體實現思路註意 system v IPC簡介 消息隊列、共用記憶體和信號量統稱為system v IPC(進程間通信機制),V是羅馬數字5,是UNI ...