Low-version Druid connection pool + MySQL driver 8.0 causes thread blocking and performance limitation

Low-version Druid connection pool + MySQL driver 8.0 causes thread blocking and performance limitation

Phenomenon

After 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 Analysis

public 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 time

public 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:

  • testOnBorrow: Execute validationQuery every time a connection is acquired to check whether the connection is valid (which will affect performance)
  • testOnReturn: Execute validationQuery every time a connection is returned to check whether the connection is valid (which will affect performance)
  • testWhileIdle: Check when applying for a connection. If the idle time is greater than timeBetweenEvictionRunsMillis, execute validationQuery to check whether the connection is valid.
  • The application sets testOnBorrow=true. Every time a connection is acquired, the synchronized lock is seized, so the performance is significantly reduced.

Solution

It 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:
  • Solve the compatibility issue between MySQL 8.0 driver and Alibaba Druid version
  • Notes on matching MySql 8.0 and corresponding driver packages
  • A brief analysis of the problem of Mysql 8.0 version driving getTables to return all database tables
  • Detailed explanation of the problems encountered when using the Mysql8.0 version driver in Mybatis reverse engineering

<<:  IDEA uses the Docker plug-in (novice tutorial)

>>:  Vue implements simple notepad function

Recommend

The most detailed method to install docker on CentOS 8

Install Docker on CentOS 8 Official documentation...

Solve the error "Can't locate ExtUtils/MakeMaker.pm in @INC"

When installing mha4mysql, the steps are roughly:...

CSS to achieve chat bubble effect

1. Rendering JD Effect Simulation Effect 2. Princ...

MySQL data duplicate checking and deduplication implementation statements

There is a table user, and the fields are id, nic...

How to export and import .sql files under Linux command

This article describes how to export and import ....

JavaScript common statements loop, judgment, string to number

Table of contents 1. switch 2. While Loop 3. Do/W...

Solve the Linux Tensorflow2.0 installation problem

conda update conda pip install tf-nightly-gpu-2.0...

MySQL 8.0.20 installation and configuration tutorial under Docker

Docker installs MySQL version 8.0.20 for your ref...

5 tips for writing CSS to make your style more standardized

1. Arrange CSS in alphabetical order Not in alphab...

A brief analysis of CSS :is() and :where() coming to browsers soon

Preview versions of Safari (Technology Preview 10...