IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 大数据 -> java执行sql慢 navicat不慢 见鬼了 -> 正文阅读

[大数据]java执行sql慢 navicat不慢 见鬼了

大家好,我是烤鸭:

???有点意思的问题,代码提示接口超时(10s+),接口逻辑很简单,就一个sql查询。本来也想是sql慢,可是拿sql去Navicat执行下,一点不慢(50ms)。

环境

DB:SqlServer

连接池:Druid

ORM:Mybatis

猜想

是刚好那个时段有其他操作造成的阻塞?

有特定参数造成的大量回表?

用了三方插件有bug?skywalking?pagehelper?

尝试复现

这个问题只在生产环境出现,即便是把生产数据备份到测试库,也不能复现。

可能生产库和测试库版本和配置不一样。

原sql

<if test="title != null and title!=''">
    AND vv.title  LIKE CONCAT(CONCAT('%', #{title}), '%')
</if>

同事有人说把参数写死试试,原来的sql类似这样。不慢了,什么原理。

<if test="title != null and title!=''">
    AND vv.title  LIKE CONCAT(CONCAT('%', 'abc'), '%')
</if>

再把#换成$,还是很快。大概就知道要看哪块的源码了。

<if test="title != null and title!=''">
    AND vv.title  LIKE CONCAT(CONCAT('%', '${title}'), '%')
</if>

关于mybatis的#和$,这篇文章写的挺好的,就不再重复了。

https://blog.csdn.net/weixin_43401380/article/details/122504003

源码分析

先说下sql执行的大致流程。

ORM框架(Mybatis做完动态sql之后) —> 连接池插件(Druid) —> JDBC —> DB(Sqlserver)。

下图是以查询方法为例。

在这里插入图片描述

接着我们看下源码:

SQLServerPreparedStatement.doPrepExec(执行RPC请求,构建请求sql,以及是否需要参数等)

final void doExecutePreparedStatement(PrepStmtExecCmd command) throws SQLServerException {
    resetForReexecute();

    // ...
    boolean hasExistingTypeDefinitions = preparedTypeDefinitions != null;
    boolean hasNewTypeDefinitions = true;
    if (!encryptionMetadataIsRetrieved) {
    	// 动态sql变量初始化
        hasNewTypeDefinitions = buildPreparedStrings(inOutParam, false);
    }

    // ...

    String dbName = connection.getSCatalog();
    boolean needsPrepare = true;
    // Retry execution if existing handle could not be re-used.
    for (int attempt = 1; attempt <= 2; ++attempt) {
        try {
            // 构建 TDSWriter ,指令为TDS的RPC
            TDSWriter tdsWriter = command.startRequest(TDS.PKT_RPC);
			// PrepExec 执行
            needsPrepare = doPrepExec(tdsWriter, inOutParam, hasNewTypeDefinitions, hasExistingTypeDefinitions);
			// 结果监听
            ensureExecuteResultsReader(command.startResponse(getIsResponseBufferingAdaptive()));
            startResults();
            getNextResult();
        }
        catch (SQLException e) {
            // ...
        }
        break;
    }       
}

SQLServerPreparedStatement.buildPreparedStrings(

sql替换,把 ? 转换成 @p1,@p2 这种,比如 mybatis 的转换后是这样的。

AND vv.title  LIKE CONCAT(CONCAT('%', @p1), '%')

/**
 * Determines whether the statement needs to be reprepared based on a change in any of the type definitions of any of the parameters due to
 * changes in scale, length, etc., and, if so, sets the new type definition string.
 */
private boolean buildPreparedStrings(Parameter[] params,
        boolean renewDefinition) throws SQLServerException {
    String newTypeDefinitions = buildParamTypeDefinitions(params, renewDefinition);
    if (null != preparedTypeDefinitions && newTypeDefinitions.equals(preparedTypeDefinitions))
        return false;   

    preparedTypeDefinitions = newTypeDefinitions;

    /* Replace the parameter marker '?' with the param numbers @p1, @p2 etc */
    preparedSQL = connection.replaceParameterMarkers(userSQL, params, bReturnValueSyntax);
    if (bRequestedGeneratedKeys)
        preparedSQL = preparedSQL + identityQuery;

    return true;
}

SQLServerPreparedStatement.doPrepExec(调用写SQL和写参数方法)

private boolean doPrepExec(TDSWriter tdsWriter,
        Parameter[] params,
        boolean hasNewTypeDefinitions,
        boolean hasExistingTypeDefinitions) throws SQLServerException {
    
    boolean needsPrepare = (hasNewTypeDefinitions && hasExistingTypeDefinitions) || !hasPreparedStatementHandle();

    // ...
    else {
        // Move overhead of needing to do prepare & unprepare to only use cases that need more than one execution.
        // First execution, use sp_executesql, optimizing for asumption we will not re-use statement.
        if (needsPrepare 
            && !connection.getEnablePrepareOnFirstPreparedStatementCall() 
            && !isExecutedAtLeastOnce
        ) {
        	// 第一次执行的时候
            buildExecSQLParams(tdsWriter);
            isExecutedAtLeastOnce = true;
        }
        // Second execution, use prepared statements since we seem to be re-using it.
        else if(needsPrepare)
        	// 写入sql
            buildPrepExecParams(tdsWriter);
        else
            buildExecParams(tdsWriter);
    }
	// 写入参数
    sendParamsByRPC(tdsWriter, params);

    return needsPrepare;
}

SQLServerPreparedStatement.buildPrepExecParams(通过TCP(TDS协议),写入SQL和参数定义)

private void buildPrepExecParams(TDSWriter tdsWriter) throws SQLServerException {
    if (getStatementLogger().isLoggable(java.util.logging.Level.FINE))
        getStatementLogger().fine(toString() + ": calling sp_prepexec: PreparedHandle:" + getPreparedStatementHandle() + ", SQL:" + preparedSQL);
    // ...
    tdsWriter.writeShort((short) 0xFFFF); // procedure name length -> use ProcIDs
    // 执行方法是 SP_PREPEXEC
    tdsWriter.writeShort(TDS.PROCID_SP_PREPEXEC);
    tdsWriter.writeByte((byte) 0);  // RPC procedure option 1
    tdsWriter.writeByte((byte) 0);  // RPC procedure option 2

    // <prepared handle>
    // IN (reprepare): Old handle to unprepare before repreparing
    // OUT: The newly prepared handle
    tdsWriter.writeRPCInt(null, getPreparedStatementHandle(), true);
    resetPrepStmtHandle();

    // <formal parameter defn> IN,写入参数定义,比如上面的title就是 @P0 NVARCHAR(4000)
    tdsWriter.writeRPCStringUnicode((preparedTypeDefinitions.length() > 0) ? preparedTypeDefinitions : null);

    // <stmt> IN,写入SQL
    tdsWriter.writeRPCStringUnicode(preparedSQL);
}

在这里插入图片描述

在这里插入图片描述

DTVExecuteOp.execute(执行实际参数的赋值)

void execute(DTV dtv,
        String strValue) throws SQLServerException {
    if (null != strValue && strValue.length() > DataTypes.SHORT_VARTYPE_MAX_CHARS)
        dtv.setJdbcType(JDBCType.LONGNVARCHAR);
	// 实际的参数
    setTypeDefinition(dtv);
}

在这里插入图片描述

原因猜想

分析了一大顿源码,并没有找到问题所在,同样的SQL在程序执行隔一天就不慢了,不清楚是不是还有别的什么因素影响执行计划。

  1. 连接规范不同:像数据库client的实现规范有JDBC(针对java语言的)或者ODBC,我们Java服务程序用的JDBC,像navicat这些软件用的是ODBC。导致同样的SQL执行计划不同。
  2. SqlServer-Client 版本问题,不同版本的client也有不同的优化。
  3. sp_prepare | sp_execute 带参数执行 和 sp_executesql,生成的执行计划不同。

没找到答案,问题竟然自己消失了,先蹲个点,如果再出现就再研究一下。

总结

当出现这个问题的时候,不要迷,先想想这个题干成立么。

navicat 执行很快,java程序很慢,两边执行的是同一个sql么?
原本在navicat 执行的是(不慢)

SELECT TOP 15 id FROM test where tile like '%测试%'

实际执行的应该是这个 sp_prepare 和 sp_execute的语句(待验证,由于问题消失,没法验证了)

declare @N int
exec sp_prepare @n output,N'@p1 NVARCHAR (4000)',N'SELECT TOP 15 id FROM test where tile like ''%'' + @p1 +''%'' '
exec sp_execute @n,'aaa' --@n就是sp_prepare返回的句柄,使用sp_execute来通过这个句柄来传递参数

如果执行这个 navicat 也变慢了,那就可以推论是 sp_prepare 这种方式导致的。

看了源码还是很难定位问题,越难的问题,解决起来越有意思,等再出现记录一下。

相关文章

JDBC和ODBC区别:
https://www.php.cn/mysql-tutorials-414951.html

DTS协议分析:
https://www.docin.com/p-98157348.html

Sp_prepare | sp_execute 介绍:

https://www.cnblogs.com/gered/p/14648626.html

EXEC和sp_executesql使用介绍:

https://blog.csdn.net/neweastsun/article/details/40019439

SQLServer执行动态SQL:

https://www.gxlcms.com/mssql-350221.html

SQLServer执行计划:

https://www.jianshu.com/p/172a345fee95

常见问题

必须声明标量变量:

https://blog.csdn.net/dxnn520/article/details/17304573

  大数据 最新文章
实现Kafka至少消费一次
亚马逊云科技:还在苦于ETL?Zero ETL的时代
初探MapReduce
【SpringBoot框架篇】32.基于注解+redis实现
Elasticsearch:如何减少 Elasticsearch 集
Go redis操作
Redis面试题
专题五 Redis高并发场景
基于GBase8s和Calcite的多数据源查询
Redis——底层数据结构原理
上一篇文章           查看所有文章
加:2022-05-26 15:19:52  更:2022-05-26 15:21:00 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/16 3:28:06-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码