PhenomenonAfter the application is upgraded to MySQL driver 8.0, when the concurrency is high, the monitoring points are checked and the time for the Druid connection pool to obtain the connection and execute SQL is mostly more than 200ms. When stress testing the system, it was found that a large number of threads were blocked. The thread dump information is as follows: "http-nio-5366-exec-48" #210 daemon prio=5 os_prio=0 tid=0x00000000023d0800 nid=0x3be9 waiting for monitor entry [0x00007fa4c1400000] java.lang.Thread.State: BLOCKED (on object monitor) at org.springframework.boot.web.embedded.tomcat.TomcatEmbeddedWebappClassLoader.loadClass(TomcatEmbeddedWebappClassLoader.java:66) - waiting to lock <0x0000000775af0960> (a java.lang.Object) at org.apache.catalina.loader.WebappClassLoaderBase.loadClass(WebappClassLoaderBase.java:1186) at com.alibaba.druid.util.Utils.loadClass(Utils.java:220) at com.alibaba.druid.util.MySqlUtils.getLastPacketReceivedTimeMs(MySqlUtils.java:372) Root Cause Analysispublic class MySqlUtils { public static long getLastPacketReceivedTimeMs(Connection conn) throws SQLException { if (class_connectionImpl == null && !class_connectionImpl_Error) { try { class_connectionImpl = Utils.loadClass("com.mysql.jdbc.MySQLConnection"); } catch (Throwable error) { class_connectionImpl_Error = true; } } if (class_connectionImpl == null) { return -1; } if (method_getIO == null && !method_getIO_error) { try { method_getIO = class_connectionImpl.getMethod("getIO"); } catch (Throwable error) { method_getIO_error = true; } } if (method_getIO == null) { return -1; } if (class_MysqlIO == null && !class_MysqlIO_Error) { try { class_MysqlIO = Utils.loadClass("com.mysql.jdbc.MysqlIO"); } catch (Throwable error) { class_MysqlIO_Error = true; } } if (class_MysqlIO == null) { return -1; } if (method_getLastPacketReceivedTimeMs == null && !method_getLastPacketReceivedTimeMs_error) { try { Method method = class_MysqlIO.getDeclaredMethod("getLastPacketReceivedTimeMs"); method.setAccessible(true); method_getLastPacketReceivedTimeMs = method; } catch (Throwable error) { method_getLastPacketReceivedTimeMs_error = true; } } if (method_getLastPacketReceivedTimeMs == null) { return -1; } try { Object connImpl = conn.unwrap(class_connectionImpl); if (connImpl == null) { return -1; } Object mysqlio = method_getIO.invoke(connImpl); Long ms = (Long) method_getLastPacketReceivedTimeMs.invoke(mysqlio); return ms.longValue(); } catch (IllegalArgumentException e) { throw new SQLException("getLastPacketReceivedTimeMs error", e); } catch (IllegalAccessException e) { throw new SQLException("getLastPacketReceivedTimeMs error", e); } catch (InvocationTargetException e) { throw new SQLException("getLastPacketReceivedTimeMs error", e); } } The getLastPacketReceivedTimeMs() method in MySqlUtils will load the com.mysql.jdbc.MySQLConnection class, but the class name is changed to com.mysql.cj.jdbc.ConnectionImpl in MySQL driver 8.0, so com.mysql.jdbc.MySQLConnection cannot be loaded in MySQL driver 8.0 In the implementation of the getLastPacketReceivedTimeMs() method, if Utils.loadClass("com.mysql.jdbc.MySQLConnection") fails to load the class and throws an exception, the variable class_connectionImpl_Error will be modified and the class will not be loaded the next time it is called. public class Utils { public static Class<?> loadClass(String className) { Class<?> clazz = null; if (className == null) { return null; } try { return Class.forName(className); } catch (ClassNotFoundException e) { // skip } ClassLoader ctxClassLoader = Thread.currentThread().getContextClassLoader(); if (ctxClassLoader != null) { try { clazz = ctxClassLoader.loadClass(className); } catch (ClassNotFoundException e) { // skip } } return clazz; } However, the ClassNotFoundException is also caught in the loadClass() method of Utils, which means that loadClass() will not throw an exception when it cannot load the class, which will cause the MySQLConnection class to be loaded once each time the getLastPacketReceivedTimeMs() method is called. The thread dump information shows that the thread is blocked when calling the loadClass() method of TomcatEmbeddedWebappClassLoader. public class TomcatEmbeddedWebappClassLoader extends ParallelWebappClassLoader { public Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException { synchronized (JreCompat.isGraalAvailable() ? this : getClassLoadingLock(name)) { Class<?> result = findExistingLoadedClass(name); result = (result != null) ? result : doLoadClass(name); if (result == null) { throw new ClassNotFoundException(name); } return resolveIfNecessary(result, resolve); } } This is because TomcatEmbeddedWebappClassLoader adds a synchronized lock when loading a class, which results in com.mysql.jdbc.MySQLConnection being loaded once each time the getLastPacketReceivedTimeMs() method is called, but it is never loaded. When loading a class, a synchronized lock is added, so thread blocking and performance degradation occur. getLastPacketReceivedTimeMs() method call timepublic abstract class DruidAbstractDataSource extends WrapperAdapter implements DruidAbstractDataSourceMBean, DataSource, DataSourceProxy, Serializable { protected boolean testConnectionInternal(DruidConnectionHolder holder, Connection conn) { String sqlFile = JdbcSqlStat.getContextSqlFile(); String sqlName = JdbcSqlStat.getContextSqlName(); if (sqlFile != null) { JdbcSqlStat.setContextSqlFile(null); } if (sqlName != null) { JdbcSqlStat.setContextSqlName(null); } try { if (validConnectionChecker != null) { boolean valid = validConnectionChecker.isValidConnection(conn, validationQuery, validationQueryTimeout); long currentTimeMillis = System.currentTimeMillis(); if (holder != null) { holder.lastValidTimeMillis = currentTimeMillis; holder.lastExecTimeMillis = currentTimeMillis; } if (valid && isMySql) { // unexcepted branch long lastPacketReceivedTimeMs = MySqlUtils.getLastPacketReceivedTimeMs(conn); if (lastPacketReceivedTimeMs > 0) { long mysqlIdleMillis = currentTimeMillis - lastPacketReceivedTimeMs; if (lastPacketReceivedTimeMs > 0 // && mysqlIdleMillis >= timeBetweenEvictionRunsMillis) { discardConnection(holder); String errorMsg = "discard long time none received connection." + ", jdbcUrl : " + jdbcUrl + ", jdbcUrl : " + jdbcUrl + ", lastPacketReceivedIdleMillis : " + mysqlIdleMillis; LOG.error(errorMsg); return false; } } } if (valid && onFatalError) { lock.lock(); try { if (onFatalError) { onFatalError = false; } finally lock.unlock(); } } return valid; } if (conn.isClosed()) { return false; } if (null == validationQuery) { return true; } Statement stmt = null; ResultSet rset = null; try { stmt = conn.createStatement(); if (getValidationQueryTimeout() > 0) { stmt.setQueryTimeout(validationQueryTimeout); } rset = stmt.executeQuery(validationQuery); if (!rset.next()) { return false; } finally JdbcUtils.close(rset); JdbcUtils.close(stmt); } if (onFatalError) { lock.lock(); try { if (onFatalError) { onFatalError = false; } finally lock.unlock(); } } return true; } catch (Throwable ex) { // skip return false; finally if (sqlFile != null) { JdbcSqlStat.setContextSqlFile(sqlFile); } if (sqlName != null) { JdbcSqlStat.setContextSqlName(sqlName); } } } The getLastPacketReceivedTimeMs() method is only called in the testConnectionInternal() method of DruidAbstractDataSource testConnectionInternal() is used to check whether the connection is valid. This method may be called when obtaining a connection or returning a connection, depending on the parameters that Druid uses to check whether the connection is valid. Parameters for Druid to detect whether the connection is valid:
SolutionIt has been verified that this bug occurs when using Druid 1.x version <= 1.1.22. The solution is to upgrade to Druid 1.x version >= 1.1.23 or Druid 1.2.x version GitHub issue: https://github.com/alibaba/druid/issues/3808 This is the end of this article about low-version Druid connection pool + MySQL driver 8.0 causing thread blocking and limited performance. For more information about MySQL driver 8.0 low-version Druid connection pool, please search 123WORDPRESS.COM's previous articles or continue to browse the following related articles. I hope everyone will support 123WORDPRESS.COM in the future! You may also be interested in:
|
<<: IDEA uses the Docker plug-in (novice tutorial)
>>: Vue implements simple notepad function
Introduction MySQL provides an EXPLAIN command th...
Install Docker on CentOS 8 Official documentation...
When installing mha4mysql, the steps are roughly:...
1. Rendering JD Effect Simulation Effect 2. Princ...
There is a table user, and the fields are id, nic...
This article describes how to export and import ....
Table of contents 1. switch 2. While Loop 3. Do/W...
Temporary tables and memory tables A memory table...
conda update conda pip install tf-nightly-gpu-2.0...
Docker installs MySQL version 8.0.20 for your ref...
p>Manually start in "Services" and i...
1. Create a test table CREATE TABLE `testsign` ( ...
1. Arrange CSS in alphabetical order Not in alphab...
Update: Now you can go to the MySQL official webs...
Preview versions of Safari (Technology Preview 10...