id select_type table type possible_keys key key_len ref rows Extra
1 SIMPLE t indexNULL aa 5 NULL 9 Using index
Handler_read_first 0
Handler_read_key 0
Handler_read_last 1
Handler_read_next 0
Handler_read_prev 9
Handler_read_retry 0
Handler_read_rnd 0
Handler_read_rnd_deleted 0
Handler_read_rnd_next 0
MySQL 8.0.23
explain select * from t orderby a desc;
id select_type table partitions type possible_keys key key_len ref rows filtered Extra
1 SIMPLE t NULLindexNULL ad 5 NULL 10 100.00 Using index
Handler_read_first 1
Handler_read_key 1
Handler_read_last 0
Handler_read_next 9
Handler_read_prev 0
Handler_read_rnd 0
Handler_read_rnd_next 0
ANALYZE/statistics collection doesn't change the outcome.
Same happens on MariaDB with MyISAM/Aria (ASC index is chosen), but it's not comparable to MySQL because MySQL doesn't support DESC indexes on MyISAM tables.
Attachments
Issue Links
is caused by
MDEV-13756Implement descending index: KEY (a DESC, b ASC)
Closed
relates to
MDEV-30845filesort still select when desc/asc index matches ORDER BY
But a universal fix should be cost based, with a new handler method to retrieve the cost of the reverse scan, proper estimation for index_scan_time, etc.
psergei, what do you think, shall we have a direction heuristics here until a cost based choice is implemented?
Sergei Golubchik
added a comment - We cannot do it in 10.8 properly. A simple heuristics would solve this particular case:
--- a/sql/sql_select.cc
+++ b/sql/sql_select.cc
@@ -29143,6 +29143,7 @@ test_if_cheaper_ordering(const JOIN_TAB *tab, ORDER *order, TABLE *table,
(select_limit <= MY_MIN(quick_records,best_records) ?
keyinfo->user_defined_key_parts < best_key_parts :
quick_records < best_records) ||
+ direction > best_key_direction ||
(!is_best_covering && is_covering))
{
possible_key.add("chosen", true);
But a universal fix should be cost based, with a new handler method to retrieve the cost of the reverse scan, proper estimation for index_scan_time , etc.
psergei , what do you think, shall we have a direction heuristics here until a cost based choice is implemented?
// We assume forward scan is faster than backward even if the
// key is longer. This should be taken into account in cost
// calculation.
direction > best_key_direction) {
best_key = nr;
best_key_parts = keyinfo->user_defined_key_parts;
if (saved_best_key_parts) *saved_best_key_parts = used_key_parts;
best_records = quick_records;
is_best_covering = is_covering;
best_key_direction = direction;
best_select_limit = select_limit;
Sergei Petrunia
added a comment - Note: in MySQL-8.4 it's still done with a basic heuristic:
if (best_key < 0 ||
(select_limit <= min(quick_records, best_records)
? keyinfo->user_defined_key_parts < best_key_parts
: quick_records < best_records) ||
// We assume forward scan is faster than backward even if the
// key is longer. This should be taken into account in cost
// calculation.
direction > best_key_direction) {
best_key = nr;
best_key_parts = keyinfo->user_defined_key_parts;
if (saved_best_key_parts) *saved_best_key_parts = used_key_parts;
best_records = quick_records;
is_best_covering = is_covering;
best_key_direction = direction;
best_select_limit = select_limit;
- - If there is no ref key and no usable keys has yet been found and
- there is either a group by or a FORCE_INDEX
- If the new cost is better than read_time
*/
- if (range_cost < read_time)
+ if (range_cost < read_time ||
+ (range_cost == read_time && direction > best_key_direction))
{
read_time= range_cost;
possible_key.add("chosen", true);
To integrate the direction matching in the cost based model, it seems we need a KEY_PREV_FIND_COST with a higher cost than KEY_NEXT_FIND_COST below, which will be used in the calculation when the key direction is the opposite of the order direction
Yuchen Pei
added a comment - - edited We can adapt the heuristics to the cost based model, see the following commit
upstream/bb-11.1-mdev-27419 cdeaefee21ce425211fdedd0092397ad0ac7dc2b MDEV-27419 [demo/check-ci] choose desc key for desc ordering
modified sql/sql_select.cc
@@ -32369,11 +32369,10 @@ test_if_cheaper_ordering(const JOIN_TAB *tab, ORDER *order, TABLE *table,
/*
We will try use the key if:
- - If there is no ref key and no usable keys has yet been found and
- there is either a group by or a FORCE_INDEX
- If the new cost is better than read_time
*/
- if (range_cost < read_time)
+ if (range_cost < read_time ||
+ (range_cost == read_time && direction > best_key_direction))
{
read_time= range_cost;
possible_key.add("chosen", true);
To integrate the direction matching in the cost based model, it seems we need a KEY_PREV_FIND_COST with a higher cost than KEY_NEXT_FIND_COST below, which will be used in the calculation when the key direction is the opposite of the order direction
IO_AND_CPU_COST handler::ha_keyread_time(uint index, ulong ranges,
ha_rows rows,
ulonglong blocks)
{
if (rows < ranges)
rows= ranges;
IO_AND_CPU_COST cost= keyread_time(index, ranges, rows, blocks);
cost.cpu+= ranges * KEY_LOOKUP_COST + (rows - ranges) * KEY_NEXT_FIND_COST;
return cost;
}
Here's a partial stack leading to the function:
0 in handler::ha_keyread_time of /home/ycp/source/mariadb-server/11.1/src/sql/handler.cc:3439
1 in cost_for_index_read of /home/ycp/source/mariadb-server/11.1/src/sql/sql_select.cc:8171
2 in get_range_limit_read_cost of /home/ycp/source/mariadb-server/11.1/src/sql/sql_select.cc:32034
3 in test_if_cheaper_ordering of /home/ycp/source/mariadb-server/11.1/src/sql/sql_select.cc:32355
4 in test_if_skip_sort_order of /home/ycp/source/mariadb-server/11.1/src/sql/sql_select.cc:26813
5 in JOIN::optimize_stage2 of /home/ycp/source/mariadb-server/11.1/src/sql/sql_select.cc:3433
Any thoughts monty and serg ?
People
Yuchen Pei
Elena Stepanova
Votes:
0Vote for this issue
Watchers:
4Start watching this issue
Dates
Created:
Updated:
Git Integration
Error rendering 'com.xiplink.jira.git.jira_git_plugin:git-issue-webpanel'. Please contact your Jira administrators.
{"report":{"fcp":1175.9000000953674,"ttfb":273.80000019073486,"pageVisibility":"visible","entityId":106345,"key":"jira.project.issue.view-issue","isInitial":true,"threshold":1000,"elementTimings":{},"userDeviceMemory":8,"userDeviceProcessors":64,"apdex":0.5,"journeyId":"12fd10b9-db9c-4b06-a4c0-73fa5b9495eb","navigationType":0,"readyForUser":1279.8000001907349,"redirectCount":0,"resourceLoadedEnd":859.5999999046326,"resourceLoadedStart":283.40000009536743,"resourceTiming":[{"duration":30.199999809265137,"initiatorType":"link","name":"https://jira.mariadb.org/s/2c21342762a6a02add1c328bed317ffd-CDN/lu2cib/820016/12ta74/0a8bac35585be7fc6c9cc5a0464cd4cf/_/download/contextbatch/css/_super/batch.css","startTime":283.40000009536743,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":283.40000009536743,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":313.59999990463257,"responseStart":0,"secureConnectionStart":0},{"duration":37.299999713897705,"initiatorType":"link","name":"https://jira.mariadb.org/s/7ebd35e77e471bc30ff0eba799ebc151-CDN/lu2cib/820016/12ta74/494e4c556ecbb29f90a3d3b4f09cb99c/_/download/contextbatch/css/jira.browse.project,project.issue.navigator,jira.view.issue,jira.general,jira.global,atl.general,-_super/batch.css?agile_global_admin_condition=true&jag=true&jira.create.linked.issue=true&slack-enabled=true&whisper-enabled=true","startTime":283.7000002861023,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":283.7000002861023,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":321,"responseStart":0,"secureConnectionStart":0},{"duration":432.19999980926514,"initiatorType":"script","name":"https://jira.mariadb.org/s/0917945aaa57108d00c5076fea35e069-CDN/lu2cib/820016/12ta74/0a8bac35585be7fc6c9cc5a0464cd4cf/_/download/contextbatch/js/_super/batch.js?locale=en","startTime":283.90000009536743,"connectEnd":283.90000009536743,"connectStart":283.90000009536743,"domainLookupEnd":283.90000009536743,"domainLookupStart":283.90000009536743,"fetchStart":283.90000009536743,"redirectEnd":0,"redirectStart":0,"requestStart":322.40000009536743,"responseEnd":716.0999999046326,"responseStart":390.7000002861023,"secureConnectionStart":283.90000009536743},{"duration":575.5,"initiatorType":"script","name":"https://jira.mariadb.org/s/2d8175ec2fa4c816e8023260bd8c1786-CDN/lu2cib/820016/12ta74/494e4c556ecbb29f90a3d3b4f09cb99c/_/download/contextbatch/js/jira.browse.project,project.issue.navigator,jira.view.issue,jira.general,jira.global,atl.general,-_super/batch.js?agile_global_admin_condition=true&jag=true&jira.create.linked.issue=true&locale=en&slack-enabled=true&whisper-enabled=true","startTime":284.09999990463257,"connectEnd":284.09999990463257,"connectStart":284.09999990463257,"domainLookupEnd":284.09999990463257,"domainLookupStart":284.09999990463257,"fetchStart":284.09999990463257,"redirectEnd":0,"redirectStart":0,"requestStart":322.7000002861023,"responseEnd":859.5999999046326,"responseStart":392.80000019073486,"secureConnectionStart":284.09999990463257},{"duration":113.89999961853027,"initiatorType":"script","name":"https://jira.mariadb.org/s/a9324d6758d385eb45c462685ad88f1d-CDN/lu2cib/820016/12ta74/c92c0caa9a024ae85b0ebdbed7fb4bd7/_/download/contextbatch/js/atl.global,-_super/batch.js?locale=en","startTime":284.2000002861023,"connectEnd":284.2000002861023,"connectStart":284.2000002861023,"domainLookupEnd":284.2000002861023,"domainLookupStart":284.2000002861023,"fetchStart":284.2000002861023,"redirectEnd":0,"redirectStart":0,"requestStart":322.80000019073486,"responseEnd":398.09999990463257,"responseStart":393.30000019073486,"secureConnectionStart":284.2000002861023},{"duration":114.80000019073486,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/1.0/_/download/batch/jira.webresources:calendar-en/jira.webresources:calendar-en.js","startTime":284.5,"connectEnd":284.5,"connectStart":284.5,"domainLookupEnd":284.5,"domainLookupStart":284.5,"fetchStart":284.5,"redirectEnd":0,"redirectStart":0,"requestStart":323.5,"responseEnd":399.30000019073486,"responseStart":394,"secureConnectionStart":284.5},{"duration":120,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/1.0/_/download/batch/jira.webresources:calendar-localisation-moment/jira.webresources:calendar-localisation-moment.js","startTime":284.59999990463257,"connectEnd":284.59999990463257,"connectStart":284.59999990463257,"domainLookupEnd":284.59999990463257,"domainLookupStart":284.59999990463257,"fetchStart":284.59999990463257,"redirectEnd":0,"redirectStart":0,"requestStart":323.7000002861023,"responseEnd":404.59999990463257,"responseStart":400.80000019073486,"secureConnectionStart":284.59999990463257},{"duration":36.799999713897705,"initiatorType":"link","name":"https://jira.mariadb.org/s/b04b06a02d1959df322d9cded3aeecc1-CDN/lu2cib/820016/12ta74/a2ff6aa845ffc9a1d22fe23d9ee791fc/_/download/contextbatch/css/jira.global.look-and-feel,-_super/batch.css","startTime":284.80000019073486,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":284.80000019073486,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":321.59999990463257,"responseStart":0,"secureConnectionStart":0},{"duration":124.09999990463257,"initiatorType":"script","name":"https://jira.mariadb.org/rest/api/1.0/shortcuts/820016/47140b6e0a9bc2e4913da06536125810/shortcuts.js?context=issuenavigation&context=issueaction","startTime":284.90000009536743,"connectEnd":284.90000009536743,"connectStart":284.90000009536743,"domainLookupEnd":284.90000009536743,"domainLookupStart":284.90000009536743,"fetchStart":284.90000009536743,"redirectEnd":0,"redirectStart":0,"requestStart":354.2000002861023,"responseEnd":409,"responseStart":407.30000019073486,"secureConnectionStart":284.90000009536743},{"duration":36.700000286102295,"initiatorType":"link","name":"https://jira.mariadb.org/s/3ac36323ba5e4eb0af2aa7ac7211b4bb-CDN/lu2cib/820016/12ta74/d176f0986478cc64f24226b3d20c140d/_/download/contextbatch/css/com.atlassian.jira.projects.sidebar.init,-_super,-project.issue.navigator,-jira.view.issue/batch.css?jira.create.linked.issue=true","startTime":285.09999990463257,"connectEnd":0,"connectStart":0,"domainLookupEnd":0,"domainLookupStart":0,"fetchStart":285.09999990463257,"redirectEnd":0,"redirectStart":0,"requestStart":0,"responseEnd":321.80000019073486,"responseStart":0,"secureConnectionStart":0},{"duration":130.2999997138977,"initiatorType":"script","name":"https://jira.mariadb.org/s/5d5e8fe91fbc506585e83ea3b62ccc4b-CDN/lu2cib/820016/12ta74/d176f0986478cc64f24226b3d20c140d/_/download/contextbatch/js/com.atlassian.jira.projects.sidebar.init,-_super,-project.issue.navigator,-jira.view.issue/batch.js?jira.create.linked.issue=true&locale=en","startTime":285.2000002861023,"connectEnd":285.2000002861023,"connectStart":285.2000002861023,"domainLookupEnd":285.2000002861023,"domainLookupStart":285.2000002861023,"fetchStart":285.2000002861023,"redirectEnd":0,"redirectStart":0,"requestStart":379.30000019073486,"responseEnd":415.5,"responseStart":410.40000009536743,"secureConnectionStart":285.2000002861023},{"duration":565.0999999046326,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/1.0/_/download/batch/jira.webresources:bigpipe-js/jira.webresources:bigpipe-js.js","startTime":289.40000009536743,"connectEnd":289.40000009536743,"connectStart":289.40000009536743,"domainLookupEnd":289.40000009536743,"domainLookupStart":289.40000009536743,"fetchStart":289.40000009536743,"redirectEnd":0,"redirectStart":0,"requestStart":388.5,"responseEnd":854.5,"responseStart":846.4000000953674,"secureConnectionStart":289.40000009536743},{"duration":556.7000002861023,"initiatorType":"script","name":"https://jira.mariadb.org/s/d41d8cd98f00b204e9800998ecf8427e-CDN/lu2cib/820016/12ta74/1.0/_/download/batch/jira.webresources:bigpipe-init/jira.webresources:bigpipe-init.js","startTime":299,"connectEnd":299,"connectStart":299,"domainLookupEnd":299,"domainLookupStart":299,"fetchStart":299,"redirectEnd":0,"redirectStart":0,"requestStart":430.59999990463257,"responseEnd":855.7000002861023,"responseStart":848.5,"secureConnectionStart":299},{"duration":236,"initiatorType":"xmlhttprequest","name":"https://jira.mariadb.org/rest/webResources/1.0/resources","startTime":894.4000000953674,"connectEnd":894.4000000953674,"connectStart":894.4000000953674,"domainLookupEnd":894.4000000953674,"domainLookupStart":894.4000000953674,"fetchStart":894.4000000953674,"redirectEnd":0,"redirectStart":0,"requestStart":1094.7000002861023,"responseEnd":1130.4000000953674,"responseStart":1128.8000001907349,"secureConnectionStart":894.4000000953674}],"fetchStart":0,"domainLookupStart":0,"domainLookupEnd":0,"connectStart":0,"connectEnd":0,"requestStart":59,"responseStart":274,"responseEnd":302,"domLoading":279,"domInteractive":1449,"domContentLoadedEventStart":1449,"domContentLoadedEventEnd":1525,"domComplete":2128,"loadEventStart":2128,"loadEventEnd":2129,"userAgent":"Mozilla/5.0 AppleWebKit/537.36 (KHTML, like Gecko; compatible; ClaudeBot/1.0; +claudebot@anthropic.com)","marks":[{"name":"bigPipe.sidebar-id.start","time":1340.9000000953674},{"name":"bigPipe.sidebar-id.end","time":1341.7000002861023},{"name":"bigPipe.activity-panel-pipe-id.start","time":1341.9000000953674},{"name":"bigPipe.activity-panel-pipe-id.end","time":1345},{"name":"activityTabFullyLoaded","time":1543.5999999046326}],"measures":[],"correlationId":"df76da050e30a8","effectiveType":"4g","downlink":10,"rtt":0,"serverDuration":110,"dbReadsTimeInMs":12,"dbConnsTimeInMs":20,"applicationHash":"9d11dbea5f4be3d4cc21f03a88dd11d8c8687422","experiments":[]}}
We cannot do it in 10.8 properly. A simple heuristics would solve this particular case:
--- a/sql/sql_select.cc
+++ b/sql/sql_select.cc
@@ -29143,6 +29143,7 @@ test_if_cheaper_ordering(const JOIN_TAB *tab, ORDER *order, TABLE *table,
(select_limit <= MY_MIN(quick_records,best_records) ?
keyinfo->user_defined_key_parts < best_key_parts :
quick_records < best_records) ||
+ direction > best_key_direction ||
(!is_best_covering && is_covering))
{
But a universal fix should be cost based, with a new handler method to retrieve the cost of the reverse scan, proper estimation for index_scan_time, etc.
psergei, what do you think, shall we have a direction heuristics here until a cost based choice is implemented?