加速mysql查询

Posted

技术标签:

【中文标题】加速mysql查询【英文标题】:to accelerate mysql query 【发布时间】:2015-07-31 08:42:42 【问题描述】:

我在 mysql 中有这个查询。此查询运行时间过长,我知道问题出在选择器(coalesce ((SELECT ...),我不知道如何通过连接加快查询速度。

我希望你们中的一些 SQL 专家能够帮助我。

SELECT
    COALESCE(
        (SELECT CONCAT(d.PRIJEVOZNIK, ' ', d.VOZAC_TRANSFER) 
         FROM dokum_zag as d
         where d.SIFKNJ='NP' and
               d.ID_VEZA=dokum_zag.ID and 
               d.korisnicko_ime=dokum_zag.korisnicko_ime 
         ),'') as PRIJEVOZNIK,
      (RELACIJA_TRANS_VOZ_TRANS) as RELACIJA_TRANS_VOZ, 
      (PRIJEVOZNIK_POVRATNI_TRANS) as PRIJEVOZNIK_POVRATNI, 
      (VAUC_KNJIZENO_TRANS) as VAUC_KNJIZENO, 
      ID_NALOGA, 
      ID_NALOGA_POV, 
      ID_VAUCHER, 
      DOLAZAK, VRIJ_TRANSFER,ODLAZAK,VRIJEME_LETA_POVRAT ,BRDOK, NOSITELJ_REZ, RELACIJA_TRANS, VOZILO_NAZIV, BROJ_NALOGA,BROJ_NAL_POV,BROJ_VAUCHER,BROJ_SOBE,VALIZN,PAX, MPIZN,ID 
FROM
    dokum_zag 
WHERE 
    korisnicko_ime = '10' and 
    ((DOLAZAK='2015-07-30') or (ODLAZAK='2015-07-30')) and 
    STORNO <> 'DA' and 
    SIFKNJ = 'TR' and 
   ((YEAR(DOLAZAK)= '2015') or (YEAR(ODLAZAK)= '2015')) 
order by 
   (CASE WHEN DOLAZAK < '2015-07-30' THEN ODLAZAK ELSE DOLAZAK END) , 
   (CASE WHEN DOLAZAK < '2015-07-30' THEN VRIJEME_LETA_POVRAT ELSE VRIJ_TRANSFER END), ID 

【问题讨论】:

【参考方案1】:

如果没有数据库结构,并且对您要提取的内容的描述,它对您的帮助有点困难。

从逻辑的角度来看,有些东西是多余的,例如

((DOLAZAK='2015-07-30') or (ODLAZAK='2015-07-30')) and 
...
((YEAR(DOLAZAK)= '2015') or (YEAR(ODLAZAK)= '2015')) 

年份部分不是必需的,因为年份是在前两个中指定的。

另一件可能让服务器发疯的事情是那个奇怪的order by 子句,因为它在记录之间发生变化(测试它在字段上固定的这个设置)。

您还可以检查是否为外部where 子句中的所有字段正确设置了索引,以及那些不是数字的,不是varchars(例如SIFKNJSTORNO 应该是char(2) )。

coalesce 部分可以通过outer join 解决,因此不会对每一行进行计算。但这取决于你想从数据库中提取什么以及如何提取......(因为该子查询在 where 部分有它自己的字段......很奇怪)

希望这能有所帮助

【讨论】:

SIFKNJ 和 STORNO 是 char(2), (YEAR(DOLAZAK)= '2015') 或 (YEAR(ODLAZAK)= '2015') 我搬走了,查询执行时间是 8.2504 秒,如果剥离(coalesce ((SELECT ...),查询执行时间0.3559秒, 表dokum_zag,有223957条记录,【参考方案2】:
INDEX(korisnicko_ime, SIFKNJ)

(以任意顺序)可能会有所帮助

将相关子查询转换为 JOIN 可能会有所帮助。

((DOLAZAK='2015-07-30') or (ODLAZAK='2015-07-30')) and 
((YEAR(DOLAZAK)= '2015') or (YEAR(ODLAZAK)= '2015')) 

有点奇怪。这可能会有所帮助:

( SELECT ... 
     AND DOLAZAK ='2015-07-30' AND ODLAZAK >= '2015-01-01'
                               AND ODLAZAK  < '2015-01-01' + INTERVAL 1 YEAR
) UNION DISTINCT
( SELECT ... 
     AND ODLAZAK ='2015-07-30' AND DOLAZAK >= '2015-01-01'
                               AND DOLAZAK  < '2015-01-01' + INTERVAL 1 YEAR
) ORDER BY ...

为了帮助重新制定,添加 2 个复合索引:

INDEX(korisnicko_ime, SIFKNJ, DOLAZAK, ODLAZAK)
INDEX(korisnicko_ime, SIFKNJ, ODLAZAK, DOLAZAK)

【讨论】:

以上是关于加速mysql查询的主要内容,如果未能解决你的问题,请参考以下文章

加速超时的 MySQL 查询

加速使用 Group By 和 Order By 的多表 Mysql 查询

MySQL如何在我的查询中加速UDF

如何在 mysql 或 php 中优化或加速我的子查询

MySQL-索引

Mysql优化查询