記一次CentOS7-MySQL排坑歷程

来源:https://www.cnblogs.com/deecyn/archive/2019/09/01/11442319.html
-Advertisement-
Play Games

https://wangde.xin/images/article/mysql/Mysql-dolphin.png ...


一、報錯及起因

今天在 CentOS7 中安裝了 mysql5.7,然後為了測試資料庫環境是否配置成功,便寫了個基於 mybatis+Spring 的 java web 程式連接操作 mysql 資料庫,於是就一些發生了令人感到很煩的報錯和故事:

當程式涉及到關於資料庫的操作如查詢、插入等操作時,首先瀏覽器訪問會很慢,進度條一直旋轉,然後頁面會報 500 錯誤:org.mybatis.spring.MyBatisSystemException: nested exception is org.apache.ibatis.exception 。然而我在 CentOS7 服務端和 Windows 本地的 Navicat 連接 mysql 都沒問題。。。

mysql海豚

二、排錯歷程

1.檢查 sql 語句

看著這似乎是 mybatis 引起的錯誤,所以先檢查 mapper 的 xml 中 sql 語句是否有錯誤,例如參數的格式轉化、resultType 為 JavaBean、resultMap 需要定義、JavaBean 和 dao 的引入等。

檢查中並沒有發現什麼問題,而且錯誤仍然存在。

2. MySql Host is blocked because of many connection errors; unblock with 'mysqladmin flush-hosts' 報錯

原因分析:

查看 tomcat 的日誌文件,發現在報錯開始部分出現了這個錯誤。經過查詢,發現這個錯誤的 原因 是:同一個 ip 在短時間內產生太多(超過 mysql 資料庫 max_connection_errors 的最大值)中斷的資料庫連接而導致的阻塞。

解決方法:

進入 CentOS7 伺服器:

方法一:提高允許的max_connection_errors數量(治標不治本):

  1. 進入 Mysql 資料庫查看 max_connection_errors: show variables like '%max_connection_errors%';
  2. 修改 max_connection_errors 的數量為 1000: set global max_connect_errors = 1000;
  3. 查看是否修改成功:show variables like '%max_connection_errors%';

方法二:使用 mysqladmin flush-hosts 命令清理一下 hosts 文件:

  1. 查找 mysqladmin 的路徑:whereis mysqladmin
  2. 執行命令,如:/usr/local/mysql5.5.35/bin/mysqladmin -uroot -pyourpwd flush-hosts

註: 方法二清理 hosts 文件,也可以直接進入 mysql 資料庫執行命令:mysql> flush hosts;

解決了 hosts 的問題後,在 tomcat 的日誌文件,發現在報錯開始部分又出現了這個錯誤:

 com.mysql.jdbc.exceptions.jdbc4.CommunicationsException: Communications link failure

The last packet successfully received from the server was 32 milliseconds ago.  The last packet sent successfully to the server was 32 milliseconds ago.
    at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
    at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:62)
    at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:45)
    at java.lang.reflect.Constructor.newInstance(Constructor.java:423)
    at com.mysql.jdbc.Util.handleNewInstance(Util.java:425)
    at com.mysql.jdbc.SQLError.createCommunicationsException(SQLError.java:990)
    ......

原因分析:

表示程式與 MySQL 通訊失敗了,即連接失敗了。The last packet successfully received from the server was 32 milliseconds ago 表示 mysql 重連,連接丟失。此為資料庫連接空閑回收問題,程式打開資料庫連接後,等待做資料庫操作時,發現連接被 MySQL 關閉掉了。

認為可能是連接等待超時問題。在 mysql 資料庫的配置中,其連接的等待時間(wait_timeout)預設值為 8 小時。在 mysql 中可以查看:

mysql﹥ 
mysql﹥ show global variables like 'wait_timeout'; 
+---------------+---------+ 
| Variable_name | Value | 
+---------------+---------+ 
| wait_timeout | 28800 | 
+---------------+---------+ 
1 row in set (0.00 sec) 

28800 seconds,也就是8小時。如果在 wait_timeout 秒期間內,資料庫連接(java.sql.Connection)一直處於等待狀態,mysql 就將該連接關閉。這時,Java 應用的連接池仍然合法地持有該連接的引用。當用該連接來進行資料庫操作時,就碰到上述錯誤。

MySQL 連接一次連接需求會經過 6 次「握手」方可成功,任何一次「握手」失敗都可能導致連接失敗。前三次握手可以簡單理解為 TCP 建立連接所必須的三次握手,MySQL 無法控制,更多的受制於 tcp 協議的不同實現,後面三次握手過程超時與 connect_timeout 有關。

解決方法:

改變資料庫參數是最簡單的處理方式(修改 /etc/my.cnf 中的 wait_timeout 值),但是需要重啟資料庫,影響較大。在不修改資料庫參數的前提下,可以做已下處理:

  • 如果使用的是 jdbc ,在 jdbc url 上添加 autoReconnect=true ,如:dataSource.url=jdbc:mysql://132.231.xx.xxx:3306/shop?useUnicode=true&characterEncoding=utf8&useSSL=true&autoReconnect=true

  • 如果是在 Spring 中使用 DBCP 連接池,在定義 datasource 增加屬性 validationQuery 和 testOnBorrow ,如:

    <bean id="vrsRankDataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
        <property name="driverClassName" value="${dataSource.driver}"/>
        <property name="url" value="${dataSource.url}"/>
        <property name="username" value="${dataSource.user}"/>
        <property name="password" value="${dataSource.password}"/>
        <property name="validationQuery" value="SELECT 1"/>
        <property name="testOnBorrow" value="true"/>
    </bean>
  • 如果是在 Spring 中使用 c3p0 連接池,則在定義 datasource 的時候,添加屬性 testConnectionOnCheckin 和 testConnectionOnCheckout ,如:

    <bean name="cacheCloudDB" class="com.mchange.v2.c3p0.ComboPooledDataSource">
        <property name="driverClass" value="${dataSource.driver}"/>
        <property name="jdbcUrl" value="${dataSource.url}"/>
        <property name="user" value="${dataSource.user}"/>
        <property name="password" value="${dataSource.password}"/>
        <property name="initialPoolSize" value="10"/>
        <property name="maxPoolSize" value="10"/>
        <property name="testConnectionOnCheckin" value="false"/>
        <property name="testConnectionOnCheckout" value="true"/>
        <property name="preferredTestQuery" value="SELECT 1"/>
    </bean>

4. 遠程連接 Mysql 太慢問題

嘗試解決了一下上面的連接超時問題,但是發現並沒有什麼用,還是會出現上面的問題。於是便懷疑是不是遠程連接 Mysql 太慢導致了連接超時?因為我在 CentOS7 服務端和 Windows 本地的 Navicat 連接 mysql 都沒問題。在網上查詢了下,發現在 mysql 的配置文件 /etc/my.cnf 中增加如下配置參數:

# 註意該配置是加在[mysqld]下麵
[mysqld]
skip-name-resolve

然後需要重啟 mysql 服務。因為根據說明,如果 mysql 主機查詢和解析 DNS 會導致緩慢或是有很多客戶端主機時會導致連接很慢。同時,請註意在增加該配置參數後,mysql的授權表中的host欄位就不能夠使用功能變數名稱而只能夠使用ip地址了,因為這是禁止了功能變數名稱解析的結果。

5. 終極解決:Could not create connection to database server. Attempted reconnect 3 times. Giving up 報錯

原因分析:

經過上面的配置後,重新測試程式,就又出現了這個錯誤。經過查詢,發現這是由於 SSL 引起的錯誤。因為我是在 CentOS7 上使用 yum 命令新裝的 MySQL5.7,預設是沒有配置 MySQL SSL 的。

而我以前一直使用的 Ubuntu16.04 上的 mysql 資料庫,它預設是配置了 MySQL SSL 的,所以我習慣在連接資料庫的 jdbc 的 url 裡面加上 &useSSL=true ,即使用 SSL 加密。導致我在連接當前 mysql 的時候,一直連接不上。查看 tomcat 日誌的報錯末尾,會有如下報錯:

Caused by: javax.net.ssl.SSLHandshakeException: java.security.cert.CertificateException: java.security.cert.CertPathValidatorException: Path does not chain with any of the trust anchors
    at sun.security.ssl.Alerts.getSSLException(Alerts.java:192)
    at sun.security.ssl.SSLSocketImpl.fatal(SSLSocketImpl.java:1946)
    at sun.security.ssl.Handshaker.fatalSE(Handshaker.java:316)
    at sun.security.ssl.Handshaker.fatalSE(Handshaker.java:310)
    at sun.security.ssl.ClientHandshaker.serverCertificate(ClientHandshaker.java:1639)
    at sun.security.ssl.ClientHandshaker.processMessage(ClientHandshaker.java:223)
    at sun.security.ssl.Handshaker.processLoop(Handshaker.java:1037)
    at sun.security.ssl.Handshaker.process_record(Handshaker.java:965)
    at sun.security.ssl.SSLSocketImpl.readRecord(SSLSocketImpl.java:1064)
    at sun.security.ssl.SSLSocketImpl.performInitialHandshake(SSLSocketImpl.java:1367)
    at sun.security.ssl.SSLSocketImpl.startHandshake(SSLSocketImpl.java:1395)
    at sun.security.ssl.SSLSocketImpl.startHandshake(SSLSocketImpl.java:1379)
    at com.mysql.jdbc.ExportControlled.transformSocketToSSLSocket(ExportControlled.java:186)
    ... 24 more
Caused by: java.security.cert.CertificateException: java.security.cert.CertPathValidatorException: Path does not chain with any of the trust anchors
    at com.mysql.jdbc.ExportControlled$X509TrustManagerWrapper.checkServerTrusted(ExportControlled.java:302)
    at sun.security.ssl.AbstractTrustManagerWrapper.checkServerTrusted(SSLContextImpl.java:1091)
    at sun.security.ssl.ClientHandshaker.serverCertificate(ClientHandshaker.java:1621)
    ... 32 more
Caused by: java.security.cert.CertPathValidatorException: Path does not chain with any of the trust anchors
    at sun.security.provider.certpath.PKIXCertPathValidator.validate(PKIXCertPathValidator.java:154)
    at sun.security.provider.certpath.PKIXCertPathValidator.engineValidate(PKIXCertPathValidator.java:80)
    at java.security.cert.CertPathValidator.validate(CertPathValidator.java:292)
    at com.mysql.jdbc.ExportControlled$X509TrustManagerWrapper.checkServerTrusted(ExportControlled.java:295)
    ... 34 more

遂恍然大悟。。。

解決方法:

在配置 JDBC URL 時使用 &useSSL=false ,即不使用 SSL 加密,配置後連接正常:

dataSource.url = jdbc:mysql://132.231.xx.xxx:3306/shop?useUnicode=true&characterEncoding=utf8&useSSL=false&autoReconnect=true

如果安全性較高的數據建議在 MySQL 端還是把 SSL 配上。

三、總結

由於 MySQL 的 SSL 問題引起了一連串的問題。由於 SSL 加密的問題,導致程式向 mysql 的連接超時,然後一直向 mysql 發送連接,導致了同一個 ip 在短時間內產生太多中斷的資料庫連接而導致的阻塞。

以後看報錯日誌,除了看報錯的最開始部分,也要看看報錯的結尾部分,弄不好會有一些其他的發現的。


PS:如果覺得文章有什麼地方寫錯了,哪裡寫得不好,或者有什麼建議,歡迎指點。

歡迎您的點贊、收藏和評論!
(完)


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

-Advertisement-
Play Games
更多相關文章
  • 一. LVS概述 LVS是一種工作在四層協議上的負載均衡解決方案,在1998年5月由章文嵩博士創建。目前廣泛使用的負載均衡模型主要有: 1)工作在四層協議(LVS):主要用於四層協議上的負載均衡,性能相較於工作在七層的協議更好。但是協議棧是工作在傳輸層,對於傳輸層以上的高級特性支持不足。 2)工作在 ...
  • 卸載Hyper-V,然後重裝,再重啟已有的Hyper-V伺服器,報錯如下: 嘗試啟動選定的虛擬機時出錯。“SP2019SER”無法更改狀態。 原因:卸載後導致虛擬網卡出現問題導致的。 解決辦法: 右擊目標虛擬機 設置 網路適配器,提示有配置錯誤,將其修正即可。 ...
  • 這兩天遇到一個頭疼的問題,我們系統需要請求第三方數據,第三方收到請求後會生成相應的數據併入庫,我們通過定時任務將第三方數據同步到我們資料庫。當我們發送請求後第三方會立即返回一個值,我們會根據返回值去資料庫更新同步過來的表欄位,sql語句執行完了,沒有任何錯誤,在同步表中查看同步的數據都有且where ...
  • 引言 MySQL中定義數據欄位的類型對你資料庫的優化是非常重要的。 MySQL支持多種類型,大致可以分為三類:數值、日期/時間和字元串(字元)類型,如下腦圖所示: 數值類型 其中: 這些類型,是定長的,其容量是不會隨著後面的數字而變化的,比如int(11)和int(8),都是一樣的占4位元組。tiny ...
  • 某大師曾說過,像瞭解自己的老婆 一樣瞭解自己管理的資料庫,個人認為包含了兩個方面的瞭解: 1,在穩定性層面來說,更多的是關註高可用、讀寫分離、負載均衡,災備管理等等high level層面的措施(就好比要保證生活的穩定性) 2,在實例級別的來說,需要關註記憶體、IO、網路,熱點表,熱點索引,top s ...
  • [學習筆記] 結果分析:shuffle的英文是洗牌,混洗的意思,洗牌就是越亂越好的意思。當在集群的情況下是這樣的,假如有三個map節點和三個reduce節點,一號reduce節點的數據會來自於三個map節點,而不是就來自於一號map節點。所以說它們的數據會混合,路線會交叉, 3叉3。想象一下,像不像 ...
  • hadoop 開源軟體,可靠的、分散式、可伸縮的框架。 分佈在不同主機上的進程協同在一起構成的應用。 大數據解決了兩個問題 1T = 1024G1P = 1024T1E = 1024P1Z = 1024E1Y = 1024Z1N = 1024Y 1.存儲 分散式存儲 2.計算 分散式計算分散式 由分 ...
  • 1、創建資料庫 登錄MySQL服務後,使用create命令創建資料庫 也可以使用mysqladmin一步創建資料庫 2、連接資料庫 登錄MySQL服務後,使用use命令連接資料庫 3、刪除資料庫 登錄MySQL服務後,使用drop命令創建資料庫 也可以使用mysqladmin一步創建資料庫 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...