mybatis查询SqlServer慢问题及解决

 更新时间:2023年08月30日 16:35:31   作者:Ikarosxx  
这篇文章主要介绍了mybatis查询SqlServer慢问题及解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教

背景

项目中用到了mybatis连接SqlServer,但遇到了一些查询慢的问题

问题1:一样的sql,换一个日期执行计划不同

# 一样的sql,djsj不同,第一条比第二条快很多
SELECT COUNT
	( 1 ) 
FROM
	e_ld2022 a WITH ( nolock )
	INNER JOIN m_user b ON b.MID = a.dbbh 
WHERE
	a.djsj = '2022-05-18 00:00:00' 
	AND a.TQBH LIKE '01%' 
	AND a.XMBH = 41;
SELECT COUNT
	( 1 ) 
FROM
	e_ld2022 a WITH ( nolock )
	INNER JOIN m_user b ON b.MID = a.dbbh 
WHERE
	a.djsj = '2022-05-19 00:00:00' 
	AND a.TQBH LIKE '01%' 
	AND a.XMBH = 41;

解决

通过MSSM分析执行计划,发现两条查询执行计划不一致,查询慢的预估行数为1,走了LOOP,循环次数过多导致变慢

故怀疑是没有更新统计信息,导致优化器优化失败

# 更新统计信息
UPDATE STATISTICS databse.table;

执行更新统计信息后重新查询,正常。

问题2:#{}和${}导致查询效率不同

mybatis中使用到了如下语句

select a.*,x.CBX from e_ld${year} a inner join E_XM x on x.XMBH = a.XMBH
<where>
	<if test="dbbh != null and dbbh !=''">
		AND dbbh LIKE concat(#{dbbh},'%')
	</if>
	<if test="tqbh != null and tqbh !=''">
		AND TQBH LIKE concat(#{tqbh},'%')
	</if>
</where>

经过测试,只要条件增加了DBBH和TQBH,就是LIKE语句,查询就巨慢

但一样的SQL拎出来放到MSSM上执行,就很快

解决

经过偶然测试,发现如果使用AND TQBH LIKE '${tqbh}%',就很快

说明问题出现在${}与#{}的区别上

以往的认知上,只知道#{}就相当于PreparedStatement,能够防注入

${}可能出现注入风险

但不知道为啥这个会影响到查询性能

跟了一下源码,发现跟sendStringParametersAsUnicode和数据库字段类型(VARCHAR、NVARCHAR)有关系

总结来说就是

sendStringParametersAsUnicode\字段类型VARCHARNVARCHAR
true×
false×

分析

SendStringParametersAsUnicode文档

SendStringParametersAsUnicode={true | false}.

Determines whether string parameters are sent to the SQL Server databasein Unicode or in the default character encoding of the database.

True means that string parameters are sent to SQL Serverin Unicode.

False means that they are sent in the default encoding, which can improve performance because the serverdoes not need to convert Unicode characters to the default encoding.

You should, however, use default encoding only if theparameter string data that you specify is consistent with the default encoding of the database.The default is true.

如果SendStringParametersAsUnicode=true,会将String参数用Unicode的编码方式发送个服务器,即 JDBCType=NVARCHAR

如果SendStringParametersAsUnicode=false,会将String参数用默认的方式发送给服务器,即 JDBCType=VARCHAR

我们系统里 SendStringParametersAsUnicode=false ,所以指定的 JdbcType=VARCHAR

但是数据库字段是NVARCHAR,和我们指定的不一致,所以会发生转换性能损耗,导致时间查询慢

基于这个思路,我测试了一下

select a.*,x.CBX from e_ld${year} a inner join E_XM x on x.XMBH = a.XMBH
<where>
	<if test="dbbh != null and dbbh !=''">
		AND dbbh LIKE concat(#{dbbh, jdbcType=NVARCHAR},'%')
	</if>
	<if test="tqbh != null and tqbh !=''">
		AND TQBH LIKE concat(#{tqbh, jdbcType=NVARCHAR},'%')
	</if>
</where>

在mybatis xml中手动指定参数的jdbcType和数据库保持一致,和结论一样,可以正常执行,不会出现查询慢的情况。

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • java中pdf转图片的实现方法

    java中pdf转图片的实现方法

    下面小编就为大家带来一篇java中pdf转图片的实现方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-12-12
  • Spring Security OAuth2 实现登录互踢的示例代码

    Spring Security OAuth2 实现登录互踢的示例代码

    这篇文章主要介绍了Spring Security OAuth2实现登录互踢的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-04-04
  • Java中的 HTTP 协议原理详解

    Java中的 HTTP 协议原理详解

    这篇文章主要介绍了Java中的 HTTP 协议原理详解,HTTP超文本传输协议,下文简称 HTTP,它的作用是用于实现服务器端和客户端的数据传输的
    2022-07-07
  • Java 获取Web项目相对webapp地址的实例

    Java 获取Web项目相对webapp地址的实例

    下面小编就为大家带来一篇Java 获取Web项目相对webapp地址的实例。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-11-11
  • Spring中@Async注解执行异步任务的方法

    Spring中@Async注解执行异步任务的方法

    在业务处理中,有些业务使用异步的方式更为合理,这篇文章主要介绍了Spring中@Async注解执行异步任务的方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-06-06
  • java并发编程实例分析

    java并发编程实例分析

    在本文里我们给大家分享了关于java并发编程实例分析以及相关知识点,需要的朋友们学习下。
    2019-03-03
  • Java实现获取客户端真实IP方法小结

    Java实现获取客户端真实IP方法小结

    本文给大家汇总介绍了2种使用java实现获取客户端真实IP的方法,主要用于获取使用了代理访问的来访者的IP,有需要的小伙伴可以参考下。
    2016-03-03
  • SpringBoot实现多数据源的实战案例

    SpringBoot实现多数据源的实战案例

    这篇文章主要介绍了SpringBoot实现多数据源的实战案例,文中通过示例代码和图文展示介绍的非常详细,对大家的学习或工作有一定的参考学习价值,需要的朋友们下面随着小编来一起学习吧
    2024-01-01
  • Mac M1安装JDK的实战避坑指南

    Mac M1安装JDK的实战避坑指南

    这篇文章主要给大家介绍了关于Mac M1安装JDK避坑的相关资料,文中通过图文介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2023-02-02
  • Java代码实践12306售票算法(二)

    Java代码实践12306售票算法(二)

    这篇文章主要介绍了Java代码实践12306售票算法(二)的相关资料,需要的朋友可以参考下
    2016-02-02

最新评论