结果集不稳定

    出于便捷的考量,MySQL “扩展” 了 group by 语法,使 select 子句可以引用未在 group by 子句中声明的非聚集字段,也就是 non-full group by 语法。在其他数据库中,这被认为是一种语法错误,因为这会导致结果集不稳定。

    在下例的 3 条 SQL 语句中,第一条 SQL 使用了 full group by 语法,所有在 select 子句中引用的字段,都在 group by 子句中有所声明,所以它的结果集是稳定的,可以看到 class 与 stuname 的全部组合共有三种;第二条与第三条是同一个 SQL,但它在两次执行时得到了不同的结果,这条 SQL 的 group by 子句中仅声明了一个 class 字段,因此结果集只会针对 class 进行聚集,class 的唯一值有两个,也就是说结果集中只会包含两行数据,而 class 与 stuname 的全部组合共有三种,班级 2018_CS_03 有两位同学,每次执行时返回哪位同学是没有语义上的限制的,都是符合语义的结果。

    因此,想保障 group by 语句结果集的稳定,请使用 full group by 语法。

    MySQL 提供了一个 SQL_MODE 开关 ONLY_FULL_GROUP_BY 来控制是否进行 full group by 语法的检查,TiDB 也兼容了这个 SQL_MODE 开关:

    1. +------------+--------------+------------------+
    2. | class | stuname | max(b.courscore) |
    3. +------------+--------------+------------------+
    4. | 2018_CS_01 | MonkeyDLuffy | 95.5 |
    5. | 2018_CS_03 | PatrickStar | 99.0 |
    6. 2 rows in set (0.01 sec)
    7. Query OK, 0 rows affected (0.01 sec)
    8. mysql> select a.class, a.stuname, max(b.courscore) from stu_info a join stu_score b on a.stuno=b.stuno group by a.class order by a.class, a.stuname;
    9. ERROR 1055 (42000): Expression #2 of ORDER BY is not in GROUP BY clause and contains nonaggregated column '' which is not functionally dependent on columns in GROUP BY clause; this is incompatible with sql_mode=only_full_group_by

    运行结果简述:上例为 sql_mode 设置了 ONLY_FULL_GROUP_BY 的效果。

    在下面的案例中,用户只在 order by 子句中添加了一个字段,TiDB 只会按照这一个字段进行排序。

    当遇到相同的 order by 值时,排序结果不稳定。为减少随机性,应当尽可能保持 order by 值的唯一性。不能保证唯一的继续加,保证 order by 的字段组合是唯一时,结果才能唯一。

    结果集不稳定是因为 TiDB 是并行地从存储层读取数据,所以 group_concat() 在不加 order by 的情况下得到的结果集展现顺序容易被感知到不稳定。

    group_concat() 要获取到按顺序输出的结果集,需要把用于排序的字段添加到 order by 子句中,这样才符合 SQL 的语义。在下面的案例中,使用 group_concat() 不加 order by 的情况下拼接 customer_id,造成结果集不稳定:

    1. 不加 order by

      1. mysql> select GROUP_CONCAT( customer_id SEPARATOR ',' ) FROM customer where customer_id like '200002%';
      2. +-------------------------------------------------------------------------+
      3. | 20000200992,20000200993,20000200994,20000200995,20000200996,20000200... |
      4. +-------------------------------------------------------------------------+

      第二次查询:

    2. 加 order by

      第一次查询:

      1. mysql> select GROUP_CONCAT( customer_id order by customer_id SEPARATOR ',' ) FROM customer where customer_id like '200002%';
      2. +-------------------------------------------------------------------------+
      3. | GROUP_CONCAT(customer_id SEPARATOR ',') |
      4. +-------------------------------------------------------------------------+
      5. +-------------------------------------------------------------------------+

      第二次查询:

    返回结果与数据在存储节点 (TiKV) 上的分布有关。如果进行了多次查询,存储节点 (TiKV) 不同存储单元 (Region) 返回结果的速度不同,会造成结果不稳定。