8.2.1.19 行构造函数 table 达式优化

行构造函数允许同时比较多个值。例如,以下两个语句在语义上是等效的:

SELECT * FROM t1 WHERE (column1,column2) = (1,1);
SELECT * FROM t1 WHERE column1 = 1 AND column2 = 1;

另外,优化器以相同的方式处理两个 table 达式。

如果行构造器的列不覆盖索引的前缀,则优化器不太可能使用可用索引。考虑下 table,该 table 在(c1, c2, c3)上具有主键:

CREATE TABLE t1 (
  c1 INT, c2 INT, c3 INT, c4 CHAR(100),
  PRIMARY KEY(c1,c2,c3)
);

在此查询中,WHERE子句使用索引中的所有列。但是,行构造器本身不包含索引前缀,因此优化器仅使用c1(key_len=4c1的大小):

mysql> EXPLAIN SELECT * FROM t1
       WHERE c1=1 AND (c2,c3) > (1,1)\G
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: t1
   partitions: NULL
         type: ref
possible_keys: PRIMARY
          key: PRIMARY
      key_len: 4
          ref: const
         rows: 3
     filtered: 100.00
        Extra: Using where

在这种情况下,使用等效的非构造函数 table 达式重写行构造函数 table 达式可能会导致更完整的索引使用。对于给定的查询,行构造函数和等效的非构造函数 table 达式为:

(c2,c3) > (1,1)
c2 > 1 OR ((c2 = 1) AND (c3 > 1))

使用索引中的所有三列(key_len=12)重写查询以使用非构造函数 table 达式会在优化器中产生结果:

mysql> EXPLAIN SELECT * FROM t1
       WHERE c1 = 1 AND (c2 > 1 OR ((c2 = 1) AND (c3 > 1)))\G
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: t1
   partitions: NULL
         type: range
possible_keys: PRIMARY
          key: PRIMARY
      key_len: 12
          ref: NULL
         rows: 3
     filtered: 100.00
        Extra: Using where

因此,为了获得更好的结果,请避免将行构造函数与AND/ORtable 达式混合使用。使用一个或另一个。

在某些情况下,优化器可以将范围访问方法应用于具有行构造函数参数的IN()table 达式。参见行构造函数 table 达式的范围优化