在数据中,子查询可以分成有依赖关系的子查询和没有依赖关系的子查询。

    有依赖关系的子查询是指该子查询的执行依赖了外部查询的“变量”,所以这种子查询通常会被计算多次。

    没有依赖关系的子查询是指该子查询的执行不依赖外部查询的“变量”, 这种子查询一般只需要计算一次。

    如下分别为没有依赖关系的子查询和有依赖关系的子查询的示例。

    1. obclient> create table t1(a int primary key, b int, c int);
    2. Query OK, 0 rows affected (0.70 sec)
    3. obclient> create table t2(a int primary key, b int, c int);
    4. Query OK, 0 rows affected (0.92 sec)
    5. -- 没有依赖关系的子查询
    6. obclient> select * from t1 where t1.a in (select t2.a from t2);
    7. Empty set (0.22 sec)
    8. -- 有依赖关系的子查询,子查询中用到了外层查询变量t1.b
    9. obclient> select * from t1 where t1.a in (select t2.a from t2 where t2.b = t1.b);
    10. Empty set (0.05 sec)

    在 OceanBase 数据库中,子查询是通过 SUBPLAN FILTER 算子执行的。SUBPLAN FILTER 算子会遍历外层查询中的每一行,对于遍历的每一行都会去检查相应的子查询来判断是否满足条件,这有点类似于 NESTED LOOP JOIN。为了高效地执行子查询,满足特定条件的子查询会被改写成联接语句 NESTED LOOP JOIN,还可以选择 HASH JOIN 和 MERGE JOIN )。在 OceanBase 数据库中,没有依赖关系的子查询都会被改写成联接语句。对于有依赖关系的子查询,只有满足特定条件才会被改写成联接语句。

    如下分别为被改写成联接语句的子查询和没有被改写成联接语句的子查询的示例。

    1. obclient> create table t1(a int primary key, b int, c int);
    2. Query OK, 0 rows affected (0.70 sec)
    3. obclient> create table t2(a int primary key, b int, c int);
    4. Query OK, 0 rows affected (0.92 sec)
    5. -- 有依赖关系的子查询被改写成了semi-join,并且使用了hash semi-join来实现
    6. obclient> explain select * from t1 where t1.a in (select t2.c from t2 where t2.b = t1.b);
    7. | =======================================
    8. |ID|OPERATOR |NAME|EST. ROWS|COST|
    9. ---------------------------------------
    10. |0 |HASH SEMI JOIN| |1 |2924|
    11. |1 | TABLE SCAN |t1 |1000 |455 |
    12. |2 | TABLE SCAN |t2 |1000 |455 |
    13. =======================================
    14. Outputs & filters:
    15. -------------------------------------
    16. 0 - output([t1.a], [t1.b], [t1.c]), filter(nil),
    17. equal_conds([t1.a = t2.c], [t2.b = t1.b]), other_conds(nil)
    18. 1 - output([t1.b], [t1.a], [t1.c]), filter(nil),
    19. access([t1.b], [t1.a], [t1.c]), partitions(p0)
    20. 2 - output([t2.b], [t2.c]), filter(nil),
    21. access([t2.b], [t2.c]), partitions(p0)
    22. -- 有依赖关系的子查询不能被改写成semi-join,使用了subplan filter来实现
    23. obclient> explain select * from t1 where t1.a > (select sum(t2.c) from t2 where t2.b = t1.b);
    24. |ID|OPERATOR |NAME|EST. ROWS|COST |
    25. -------------------------------------------
    26. |0 |SUBPLAN FILTER | |334 |207683|
    27. |1 | TABLE SCAN |t1 |334 |176 |
    28. |2 | SCALAR GROUP BY| |1 |623 |
    29. |3 | TABLE SCAN |t2 |2 |622 |
    30. ===========================================
    31. Outputs & filters:
    32. -------------------------------------
    33. 0 - output([t1.a], [t1.b], [t1.c]), filter([t1.a > subquery(1)]),
    34. exec_params_([t1.b]), onetime_exprs_(nil), init_plan_idxs_(nil)
    35. 1 - output([t1.b], [t1.a], [t1.c]), filter(nil),
    36. access([t1.b], [t1.a], [t1.c]), partitions(p0)
    37. 2 - output([T_FUN_SUM(t2.c)]), filter(nil),
    38. group(nil), agg_func([T_FUN_SUM(t2.c)])
    39. 3 - output([t2.c]), filter([t2.b = ?]),
    40. access([t2.b], [t2.c]), partitions(p0)