mirror of
https://github.com/MariaDB/server.git
synced 2025-01-27 17:33:44 +01:00
512 lines
15 KiB
Text
512 lines
15 KiB
Text
#
|
|
# MDEV-34720: Poor plan choice for large JOIN with ORDER BY and small LIMIT
|
|
#
|
|
create table t1 (
|
|
a int,
|
|
b int,
|
|
c int,
|
|
col1 int,
|
|
col2 int,
|
|
index(a),
|
|
index(b),
|
|
index(col1)
|
|
);
|
|
insert into t1 select
|
|
mod(seq, 100),
|
|
mod(seq, 95),
|
|
seq,
|
|
seq,
|
|
seq
|
|
from
|
|
seq_1_to_10000;
|
|
create table t10 (
|
|
a int,
|
|
a_value char(10),
|
|
key(a)
|
|
);
|
|
insert into t10 select seq, seq from seq_1_to_100;
|
|
create table t11 (
|
|
b int,
|
|
b_value char(10),
|
|
key(b)
|
|
);
|
|
insert into t11 select seq, seq from seq_1_to_100;
|
|
set @tmp_os=@@optimizer_trace;
|
|
set optimizer_trace=1;
|
|
#
|
|
# Query 1 - basic example.
|
|
#
|
|
# Table t1 is not the first, have to use temporary+filesort:
|
|
explain
|
|
select
|
|
*
|
|
from
|
|
t1
|
|
join t10 on t1.a=t10.a
|
|
join t11 on t1.b=t11.b
|
|
order by
|
|
t1.col1
|
|
limit 10;
|
|
id select_type table type possible_keys key key_len ref rows Extra
|
|
1 SIMPLE t10 ALL a NULL NULL NULL 100 Using where; Using temporary; Using filesort
|
|
1 SIMPLE t1 ref a,b a 5 test.t10.a 100 Using where
|
|
1 SIMPLE t11 ref b b 5 test.t1.b 1
|
|
set optimizer_join_limit_pref_ratio=10;
|
|
# t1 is first, key=col1 produces ordering, no filesort or temporary:
|
|
explain
|
|
select
|
|
*
|
|
from
|
|
t1
|
|
join t10 on t1.a=t10.a
|
|
join t11 on t1.b=t11.b
|
|
order by
|
|
t1.col1
|
|
limit 10;
|
|
id select_type table type possible_keys key key_len ref rows Extra
|
|
1 SIMPLE t1 index a,b col1 5 NULL 10 Using where
|
|
1 SIMPLE t10 ref a a 5 test.t1.a 1
|
|
1 SIMPLE t11 ref b b 5 test.t1.b 1
|
|
set @trace=(select trace from information_schema.optimizer_trace);
|
|
select json_detailed(json_extract(@trace, '$**.join_limit_shortcut_choice')) as JS;
|
|
JS
|
|
[
|
|
{
|
|
"limit_fraction": 0.001,
|
|
"test_if_skip_sort_order_early":
|
|
[
|
|
{
|
|
"reconsidering_access_paths_for_index_ordering":
|
|
{
|
|
"clause": "ORDER BY",
|
|
"fanout": 1,
|
|
"read_time": 53.27053125,
|
|
"table": "t1",
|
|
"rows_estimation": 10000,
|
|
"possible_keys":
|
|
[
|
|
{
|
|
"index": "a",
|
|
"can_resolve_order": false,
|
|
"cause": "not usable index for the query"
|
|
},
|
|
{
|
|
"index": "b",
|
|
"can_resolve_order": false,
|
|
"cause": "not usable index for the query"
|
|
},
|
|
{
|
|
"index": "col1",
|
|
"can_resolve_order": true,
|
|
"direction": 1,
|
|
"updated_limit": 10,
|
|
"index_scan_time": 10,
|
|
"records": 10000,
|
|
"chosen": true
|
|
}
|
|
]
|
|
}
|
|
}
|
|
],
|
|
"can_skip_filesort": true,
|
|
"full_join_cost": "REPLACED",
|
|
"risk_ratio": 10,
|
|
"shortcut_join_cost": "REPLACED",
|
|
"shortcut_cost_with_risk": 972.8224714,
|
|
"use_shortcut_cost": true
|
|
}
|
|
]
|
|
#
|
|
# Query 2 - same as above but without a suitable index.
|
|
#
|
|
# Table t1 is not the first, have to use temporary+filesort:
|
|
set optimizer_join_limit_pref_ratio=0;
|
|
explain
|
|
select
|
|
*
|
|
from
|
|
t1
|
|
join t10 on t1.a=t10.a
|
|
join t11 on t1.b=t11.b
|
|
order by
|
|
t1.col2
|
|
limit 10;
|
|
id select_type table type possible_keys key key_len ref rows Extra
|
|
1 SIMPLE t10 ALL a NULL NULL NULL 100 Using where; Using temporary; Using filesort
|
|
1 SIMPLE t1 ref a,b a 5 test.t10.a 100 Using where
|
|
1 SIMPLE t11 ref b b 5 test.t1.b 1
|
|
# t1 is first but there's no suitable index,
|
|
# so we use filesort but using temporary:
|
|
set optimizer_join_limit_pref_ratio=10;
|
|
explain
|
|
select
|
|
*
|
|
from
|
|
t1
|
|
join t10 on t1.a=t10.a
|
|
join t11 on t1.b=t11.b
|
|
order by
|
|
t1.col2
|
|
limit 10;
|
|
id select_type table type possible_keys key key_len ref rows Extra
|
|
1 SIMPLE t1 ALL a,b NULL NULL NULL 10000 Using where; Using filesort
|
|
1 SIMPLE t10 ref a a 5 test.t1.a 1
|
|
1 SIMPLE t11 ref b b 5 test.t1.b 1
|
|
set @trace=(select trace from information_schema.optimizer_trace);
|
|
select json_detailed(json_extract(@trace, '$**.join_limit_shortcut_choice')) as JS;
|
|
JS
|
|
[
|
|
{
|
|
"limit_fraction": 0.001,
|
|
"test_if_skip_sort_order_early":
|
|
[],
|
|
"can_skip_filesort": false,
|
|
"full_join_cost": "REPLACED",
|
|
"risk_ratio": 10,
|
|
"shortcut_join_cost": "REPLACED",
|
|
"shortcut_cost_with_risk": 20972.81247,
|
|
"use_shortcut_cost": true
|
|
}
|
|
]
|
|
#
|
|
# Query 3: Counter example with large limit
|
|
#
|
|
# Table t1 is not the first, have to use temporary+filesort:
|
|
set optimizer_join_limit_pref_ratio=0;
|
|
explain
|
|
select
|
|
*
|
|
from
|
|
t1
|
|
join t10 on t1.a=t10.a
|
|
join t11 on t1.b=t11.b
|
|
order by
|
|
t1.col1
|
|
limit 5000;
|
|
id select_type table type possible_keys key key_len ref rows Extra
|
|
1 SIMPLE t10 ALL a NULL NULL NULL 100 Using where; Using temporary; Using filesort
|
|
1 SIMPLE t1 ref a,b a 5 test.t10.a 100 Using where
|
|
1 SIMPLE t11 ref b b 5 test.t1.b 1
|
|
# Same plan as above:
|
|
# Table t1 is not the first, have to use temporary+filesort:
|
|
set optimizer_join_limit_pref_ratio=10;
|
|
explain
|
|
select
|
|
*
|
|
from
|
|
t1
|
|
join t10 on t1.a=t10.a
|
|
join t11 on t1.b=t11.b
|
|
order by
|
|
t1.col1
|
|
limit 5000;
|
|
id select_type table type possible_keys key key_len ref rows Extra
|
|
1 SIMPLE t10 ALL a NULL NULL NULL 100 Using where; Using temporary; Using filesort
|
|
1 SIMPLE t1 ref a,b a 5 test.t10.a 100 Using where
|
|
1 SIMPLE t11 ref b b 5 test.t1.b 1
|
|
set @trace=(select trace from information_schema.optimizer_trace);
|
|
select json_detailed(json_extract(@trace, '$**.join_limit_shortcut_choice')) as JS;
|
|
JS
|
|
[
|
|
{
|
|
"limit_fraction": 0.5,
|
|
"test_if_skip_sort_order_early":
|
|
[
|
|
{
|
|
"reconsidering_access_paths_for_index_ordering":
|
|
{
|
|
"clause": "ORDER BY",
|
|
"fanout": 1,
|
|
"read_time": 53.27053125,
|
|
"table": "t1",
|
|
"rows_estimation": 10000,
|
|
"possible_keys":
|
|
[
|
|
{
|
|
"index": "a",
|
|
"can_resolve_order": false,
|
|
"cause": "not usable index for the query"
|
|
},
|
|
{
|
|
"index": "b",
|
|
"can_resolve_order": false,
|
|
"cause": "not usable index for the query"
|
|
},
|
|
{
|
|
"index": "col1",
|
|
"can_resolve_order": true,
|
|
"direction": 1,
|
|
"updated_limit": 5000,
|
|
"index_scan_time": 5000,
|
|
"usable": false,
|
|
"cause": "cost"
|
|
}
|
|
]
|
|
}
|
|
}
|
|
],
|
|
"can_skip_filesort": false,
|
|
"full_join_cost": "REPLACED",
|
|
"risk_ratio": 10,
|
|
"shortcut_join_cost": "REPLACED",
|
|
"shortcut_cost_with_risk": 240591.2748,
|
|
"use_shortcut_cost": false
|
|
}
|
|
]
|
|
#
|
|
# Query 4: LEFT JOIN makes it impossible to put ORDER-BY-table first,
|
|
# however the optimizer still puts it as sort_by_table.
|
|
#
|
|
set optimizer_join_limit_pref_ratio=10;
|
|
explain
|
|
select
|
|
*
|
|
from
|
|
t10 left join (t1 join t11 on t1.b=t11.b ) on t1.a=t10.a
|
|
order by
|
|
t1.col2
|
|
limit 10;
|
|
id select_type table type possible_keys key key_len ref rows Extra
|
|
1 SIMPLE t10 ALL NULL NULL NULL NULL 100 Using temporary; Using filesort
|
|
1 SIMPLE t1 ref a,b a 5 test.t10.a 100 Using where
|
|
1 SIMPLE t11 ref b b 5 test.t1.b 1
|
|
set @trace=(select trace from information_schema.optimizer_trace);
|
|
# This will show nothing as limit shortcut code figures that
|
|
# it's not possible to use t1 to construct shortcuts:
|
|
select json_detailed(json_extract(@trace, '$**.join_limit_shortcut_choice')) as JS;
|
|
JS
|
|
NULL
|
|
#
|
|
# Query 5: Same as Q1 but also with a semi-join
|
|
#
|
|
set optimizer_join_limit_pref_ratio=default;
|
|
# Table t1 is not the first, have to use temporary+filesort:
|
|
explain
|
|
select
|
|
*
|
|
from
|
|
t1
|
|
join t10 on t1.a=t10.a
|
|
join t11 on t1.b=t11.b
|
|
where
|
|
t1.a in (select a from t10) and
|
|
t1.b in (select b from t11)
|
|
order by
|
|
t1.col1
|
|
limit 10;
|
|
id select_type table type possible_keys key key_len ref rows Extra
|
|
1 PRIMARY t10 ALL a NULL NULL NULL 100 Using where; Using temporary; Using filesort
|
|
1 PRIMARY t10 ref a a 5 test.t10.a 1 Using index; LooseScan
|
|
1 PRIMARY t1 ref a,b a 5 test.t10.a 100 Using where
|
|
1 PRIMARY t11 ref b b 5 test.t1.b 1
|
|
1 PRIMARY <subquery3> eq_ref distinct_key distinct_key 4 func 1
|
|
3 MATERIALIZED t11 index b b 5 NULL 100 Using index
|
|
set optimizer_join_limit_pref_ratio=10;
|
|
# t1 is first, key=col1 produces ordering, no filesort or temporary:
|
|
explain
|
|
select
|
|
*
|
|
from
|
|
t1
|
|
join t10 on t1.a=t10.a
|
|
join t11 on t1.b=t11.b
|
|
where
|
|
t1.a in (select a from t10) and
|
|
t1.b in (select b from t11)
|
|
order by
|
|
t1.col1
|
|
limit 10;
|
|
id select_type table type possible_keys key key_len ref rows Extra
|
|
1 PRIMARY t1 index a,b col1 5 NULL 1 Using where
|
|
1 PRIMARY t10 ref a a 5 test.t1.a 1
|
|
1 PRIMARY t11 ref b b 5 test.t1.b 1
|
|
1 PRIMARY <subquery2> eq_ref distinct_key distinct_key 4 func 1
|
|
1 PRIMARY <subquery3> eq_ref distinct_key distinct_key 4 func 1
|
|
2 MATERIALIZED t10 index a a 5 NULL 100 Using index
|
|
3 MATERIALIZED t11 index b b 5 NULL 100 Using index
|
|
set @trace=(select trace from information_schema.optimizer_trace);
|
|
select json_detailed(json_extract(@trace, '$**.join_limit_shortcut_choice')) as JS;
|
|
JS
|
|
[
|
|
{
|
|
"limit_fraction": 0.001,
|
|
"test_if_skip_sort_order_early":
|
|
[
|
|
{
|
|
"reconsidering_access_paths_for_index_ordering":
|
|
{
|
|
"clause": "ORDER BY",
|
|
"fanout": 1,
|
|
"read_time": 53.27053125,
|
|
"table": "t1",
|
|
"rows_estimation": 10000,
|
|
"possible_keys":
|
|
[
|
|
{
|
|
"index": "a",
|
|
"can_resolve_order": false,
|
|
"cause": "not usable index for the query"
|
|
},
|
|
{
|
|
"index": "b",
|
|
"can_resolve_order": false,
|
|
"cause": "not usable index for the query"
|
|
},
|
|
{
|
|
"index": "col1",
|
|
"can_resolve_order": true,
|
|
"direction": 1,
|
|
"updated_limit": 10,
|
|
"index_scan_time": 10,
|
|
"records": 10000,
|
|
"chosen": true
|
|
}
|
|
]
|
|
}
|
|
}
|
|
],
|
|
"can_skip_filesort": true,
|
|
"full_join_cost": "REPLACED",
|
|
"risk_ratio": 10,
|
|
"shortcut_join_cost": "REPLACED",
|
|
"shortcut_cost_with_risk": 982.9697956,
|
|
"use_shortcut_cost": true
|
|
}
|
|
]
|
|
#
|
|
# Query 6: same as Query 1 but let's limit the search depth
|
|
#
|
|
set @tmp_osd=@@optimizer_search_depth;
|
|
set optimizer_search_depth=1;
|
|
set optimizer_join_limit_pref_ratio=default;
|
|
# Table t1 is not the first, have to use temporary+filesort:
|
|
explain
|
|
select
|
|
*
|
|
from
|
|
t1
|
|
join t10 on t1.a=t10.a
|
|
join t11 on t1.b=t11.b
|
|
order by
|
|
t1.col1
|
|
limit 10;
|
|
id select_type table type possible_keys key key_len ref rows Extra
|
|
1 SIMPLE t10 ALL a NULL NULL NULL 100 Using where; Using temporary; Using filesort
|
|
1 SIMPLE t11 ALL b NULL NULL NULL 100 Using join buffer (flat, BNL join)
|
|
1 SIMPLE t1 ref a,b a 5 test.t10.a 100 Using where
|
|
set optimizer_join_limit_pref_ratio=10;
|
|
# t1 is first, key=col1 produces ordering, no filesort or temporary:
|
|
explain
|
|
select
|
|
*
|
|
from
|
|
t1
|
|
join t10 on t1.a=t10.a
|
|
join t11 on t1.b=t11.b
|
|
order by
|
|
t1.col1
|
|
limit 10;
|
|
id select_type table type possible_keys key key_len ref rows Extra
|
|
1 SIMPLE t1 index a,b col1 5 NULL 10 Using where
|
|
1 SIMPLE t10 ref a a 5 test.t1.a 1
|
|
1 SIMPLE t11 ref b b 5 test.t1.b 1
|
|
set @trace=(select trace from information_schema.optimizer_trace);
|
|
select json_detailed(json_extract(@trace, '$**.join_limit_shortcut_choice')) as JS;
|
|
JS
|
|
[
|
|
{
|
|
"limit_fraction": 0.001,
|
|
"test_if_skip_sort_order_early":
|
|
[
|
|
{
|
|
"reconsidering_access_paths_for_index_ordering":
|
|
{
|
|
"clause": "ORDER BY",
|
|
"fanout": 1,
|
|
"read_time": 53.27053125,
|
|
"table": "t1",
|
|
"rows_estimation": 10000,
|
|
"possible_keys":
|
|
[
|
|
{
|
|
"index": "a",
|
|
"can_resolve_order": false,
|
|
"cause": "not usable index for the query"
|
|
},
|
|
{
|
|
"index": "b",
|
|
"can_resolve_order": false,
|
|
"cause": "not usable index for the query"
|
|
},
|
|
{
|
|
"index": "col1",
|
|
"can_resolve_order": true,
|
|
"direction": 1,
|
|
"updated_limit": 10,
|
|
"index_scan_time": 10,
|
|
"records": 10000,
|
|
"chosen": true
|
|
}
|
|
]
|
|
}
|
|
}
|
|
],
|
|
"can_skip_filesort": true,
|
|
"full_join_cost": "REPLACED",
|
|
"risk_ratio": 10,
|
|
"shortcut_join_cost": "REPLACED",
|
|
"shortcut_cost_with_risk": 972.8224714,
|
|
"use_shortcut_cost": true
|
|
}
|
|
]
|
|
set optimizer_search_depth=@tmp_osd;
|
|
set optimizer_trace=@tmp_os;
|
|
# An extra testcase for MDEV-35164 (its main testcase is below).
|
|
alter table t1 add unique key(col2);
|
|
insert into t10 select * from t10;
|
|
insert into t10 select * from t10;
|
|
analyze table t10;
|
|
Table Op Msg_type Msg_text
|
|
test.t10 analyze status Engine-independent statistics collected
|
|
test.t10 analyze status OK
|
|
# This will not crash and also show that sorting is not done when
|
|
# ORDER BY only refers to const table columns:
|
|
explain
|
|
select
|
|
*
|
|
from
|
|
t1
|
|
join t10 on t1.a=t10.a
|
|
join t11 on t1.b=t11.b
|
|
where
|
|
t1.col2=3
|
|
order by
|
|
t1.col1
|
|
limit 10;
|
|
id select_type table type possible_keys key key_len ref rows Extra
|
|
1 SIMPLE t1 const col2,a,b col2 5 const 1
|
|
1 SIMPLE t11 ref b b 5 const 1
|
|
1 SIMPLE t10 ref a a 5 const 2
|
|
drop table t1, t10, t11;
|
|
#
|
|
# MDEV-35072: Assertion failure with optimizer_join_limit_pref_ratio and 1-table select
|
|
#
|
|
SET optimizer_join_limit_pref_ratio=1;
|
|
CREATE TABLE t1 (c1 INT, INDEX(c1));
|
|
INSERT INTO t1 VALUES (1),(2);
|
|
SELECT * FROM t1 ORDER BY c1 LIMIT 1;
|
|
c1
|
|
1
|
|
DROP TABLE t1;
|
|
#
|
|
# MDEV-35164: optimizer_join_limit_pref_ratio: assertion when the ORDER BY table becomes constant
|
|
# Original testcase:
|
|
#
|
|
SET optimizer_join_limit_pref_ratio=1;
|
|
CREATE TABLE t1 (a INT KEY,b INT, KEY(b)) ;
|
|
INSERT INTO t1 VALUES (2,NULL);
|
|
INSERT INTO t1 VALUES (5,NULL);
|
|
SELECT * FROM t1 NATURAL JOIN t1 AS t2 WHERE t1.b=NULL ORDER BY t1.a LIMIT 1;
|
|
a b
|
|
DROP TABLE t1;
|
|
set optimizer_join_limit_pref_ratio=default;
|