CREATE TABLE `words` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`word` varchar(64) DEFAULT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB;
delimiter ;;
create procedure idata()
begin
declare i int;
set i=0;
while i<10000 do
insert into words(word) values(concat(char(97 (i div 1000)), char(97 (i % 1000 div 100)), char(97 (i % 100 div 10)), char(97 (i % 10))));
set i=i 1;
end while;
end;;
delimiter ;
call idata();
如何随機取3個單詞
select word from words order by rand() limit 3;
-- 查看上面語句的執行情況
explain select word from words order by rand() limit 3;
Extra中Using temporary表示使用臨時表,Using filesort表示需要執行排序操作。
上述默認使用的臨時表是内存表,對于内存表來說,回表過程隻是簡單地根據數據行的位置直接訪問内存得到數據,并不會導緻額外的磁盤訪問,因此MySQL會在排序時會優先使用rowid排序。
上述SQL語句的執行過程如下:
MySQL8.0下慢查詢日志如下圖,掃描行數為100003行:
臨時表隻能是内存表麼?
答案是NO。那什麼時候臨時表會使用内存,什麼時候又會使用磁盤呢?該選擇主要依賴于tmp_table_size參數的控制,默認是16M,如果臨時表的大小超過了tmp_table_size,那麼内存臨時表就會轉換為磁盤臨時表。
磁盤臨時表使用的默認引擎是InnoDB,由參數default_tmp_storage_engine控制。
set tmp_table_size=1024;
set sort_buffer_size=32768;
set max_length_for_sort_data=16;
/* 打開 optimizer_trace,隻對本線程有效 */
SET optimizer_trace='enabled=on';
/* 執行語句 */
select word from words order by rand() limit 3;
/* 查看 OPTIMIZER_TRACE 輸出 */
SELECT * FROM `information_schema`.`OPTIMIZER_TRACE`\G
在我們上圖的輸出中,我們可以看出排序算法使用了優先隊列排序算法,然後是全字段排序(也就是說不用回表)。
peak_memory_used代表排序時使用到的内存,按道理應該等于sort_buffer_size指定的值,之所以不等的原因是作者本人的MySQL是8.0.12版本。
MySQL8.0.12之前,MySQL優化器會為排序直接分配sort_buffer_size指定大小的内存,但從MySQL8.0.12開始,為排序分配内存是以增量的方式進行。
,更多精彩资讯请关注tft每日頭條,我们将持续为您更新最新资讯!