总结
- 获取慢查询SQL
- 已经执行完的SQL,检查慢查询日志,日志中有执行慢的SQL
- 正在执行中的SQL,
show proccesslist;
,结果中有执行慢的SQL
慢查询日志关键参数
名称 | 解释 |
---|---|
Query_time | 查询消耗时间 |
Time | 慢查询发生时间 |
- 分析慢查询SQL
explain 慢SQL
explain关键参数
名称 | 解释 |
---|---|
key | 实际用到的索引列 |
type | 索引类型 |
extra | 额外信息 |
type部分值
名称 | 解释 |
---|---|
consts | 基于主键或唯一索引查询,最多返回一条数据,优化阶段可得到数据 |
ref | 基于普通索引的等值查询,表间等值连接 |
range | 利用索引范围查询 |
index | 全索引扫描 |
ALL | 全表操作 |
- 阿里java开发手册-泰山版,要求至少range
Extra部分值
名称 | 解释 |
---|---|
Using index | 使用覆盖索引,减少表扫描和回表 |
Using index condition | 先条件过滤索引再查询数据 |
Using filesort | 使用外部排序,非索引排序 |
Using where | 使用where条件 |
Impossible where | where总是false |
Using temporary | 使用临时表,一般发生在order by无索引列时 |
Using join buffer (Block Nested Loop) | 在进行嵌套循环连接,内表大 |
Select tables optimized away | 该查询不需要访问实际的表,而是通过优化方式直接计算出结果 |
- 优化慢SQL
准备数据库
库
CREATE DATABASE IF NOT EXISTS test_slow DEFAULT CHARACTER SET utf8mb4 DEFAULT COLLATE utf8mb4_general_ci;
use test_slow;
表
CREATE TABLE `person_info_large` (
`id` BIGINT UNSIGNED NOT NULL AUTO_INCREMENT,
`code` VARCHAR (36),
`name` VARCHAR (36),
`title` VARCHAR (72),
`location` VARCHAR (108),
PRIMARY KEY `pk_id` (`id`),
UNIQUE `uk_code` (`code`),
KEY `idx_title_location`(`title`,`location`)
) ENGINE = INNODB AUTO_INCREMENT = 1 DEFAULT CHARSET = utf8
行(java生成sql文件》导入sql文件)
package com.xcrj.gen;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.UUID;
public class Main {
public static void main(String[] args) {
generate(300 * 10000);
}
private static String rand36Str() {
long time = System.currentTimeMillis();
int random = (int) (Math.random() * Integer.MAX_VALUE);
UUID uuid = new UUID(time, random);//随机种子
return uuid.toString();
}
private static String rand36Str(int num) {
StringBuilder sb = new StringBuilder();
UUID uuid;
for (int i = 0; i < num; i++) {
uuid = UUID.randomUUID();
sb.append(uuid.toString());
}
return sb.toString();
}
private static void generate(int size) {
String row = "INSERT INTO researcher(`code`,`name`,`title`,`location`) VALUE(%s);";
// System.out.println(String.format(sql,IdWorker.getId()));
String path = "./test_slow_researcher.sql";
File file = new File(path);
if (!file.exists()) {
try {
file.createNewFile();
} catch (IOException e) {
e.printStackTrace();
}
}
try (
FileOutputStream fos = new FileOutputStream(path);
BufferedOutputStream bos = new BufferedOutputStream(fos);
) {
for (int i = 0; i < size; i++) {
StringBuilder sb = new StringBuilder();
String code = rand36Str(1);
String name = rand36Str(1);
String title = rand36Str(2);
String location = rand36Str(3);
sb.append("'").append(code).append("'").append(",")
.append("'").append(name).append("'").append(",")
.append("'").append(title).append("'").append(",")
.append("'").append(location).append("'");
bos.write(String.format(row, sb.toString()).getBytes());
if (i < size - 1) {
bos.write("\n".getBytes());
}
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
开启慢查询日志
# 检查默认值
show variables like '%quer%';
# 开启慢查询日志
set global slow_query_log=on;
# 设置慢查询阈值为1s
set global long_query_time=1;
# 查看慢查询日志路径
show global variables like 'slow_query_log_file'
# 检查设置值,若发现未生效,关闭当前会话(关闭数据库)重新打开,再检查
show variables like '%quer%';
参数 | 含义 |
---|---|
slow_query_log | 是否开启慢查询日志 |
slow_query_log_file | 慢查询日志路径 |
long_query_time | 慢查询阈值,默认10s |
慢查询测试1,已经执行完的慢查询
# 统计
SELECT count(*) FROM researcher;
# 无索引列
SELECT `name` FROM researcher ORDER BY `name` DESC;
# 有索引列
SELECT `code` FROM researcher ORDER BY `code` DESC;
# 查询慢查询日志文件地址 /var/lib/mysql/333a2bf4a87e-slow.log
show variables like '%quer%';
# 查看慢查询日志
more /var/lib/mysql/333a2bf4a87e-slow.log
# 分析SQL
explain SELECT count(*) FROM researcher;
explain SELECT `name` FROM researcher ORDER BY `name` DESC;
explain SELECT `code` FROM researcher ORDER BY `code` DESC;
SELECT count(*) FROM researcher;
- Query_time: 大于18s
SELECT
nameFROM researcher ORDER BY
name DESC;
- Query_time: 大于19s
SELECT
codeFROM researcher ORDER BY
code DESC;
- Query_time: 大于16s
explain SELECT count(*) FROM researcher;
- key=
uk_code
,可知count(*)
会使用索引 - type=index
- extra=Using index
explain SELECT
nameFROM researcher ORDER BY
name DESC;
- key=NULL
- type=ALL
- extra=Using filesort
explain SELECT
codeFROM researcher ORDER BY
code DESC;
- key=uk_code
- type=index
- extra=Using index
explain SELECT
idFROM researcher ORDER BY
id DESC;
- key=primary
- type=index
- extra=Using index
参数说明
慢查询日志部分参数
名称 | 解释 |
---|---|
Query_time | 查询消耗时间 |
Time | 慢查询发生时间 |
explain部分值
名称 | 解释 |
---|---|
key | 实际用到的索引列 |
type | 索引类型 |
extra | 额外信息 |
select_type | 查询方式 |
possible_keys | 可能用到的索引列 |
type部分值
名称 | 解释 |
---|---|
consts | 基于主键或唯一索引查询,最多返回一条数据,优化阶段可得到数据 |
ref | 基于普通索引的等值查询,表间等值连接 |
range | 利用索引范围查询 |
index | 全索引扫描 |
ALL | 全表操作 |
- 阿里java开发手册-泰山版,要求至少range
Extra部分值
名称 | 解释 |
---|---|
Using index | 使用覆盖索引,减少表扫描和回表 |
Using index condition | 先条件过滤索引再查询数据 |
Using filesort | 使用外部排序,非索引排序 |
Using where | 使用where条件 |
Impossible where | where总是false |
Using temporary | 使用临时表,一般发生在order by无索引列时 |
Using join buffer (Block Nested Loop) | 在进行嵌套循环连接,内表大 |
Select tables optimized away | 该查询不需要访问实际的表,而是通过优化方式直接计算出结果 |
select_type部分值
名称 | 解释 |
---|---|
Simple | 简单查询 |
Primary | 关联查询或子查询的外层查询 |
Unoin | 关联查询或子查询的后续查询 |
慢查询测试2,正在执行的慢查询
SELECT `name` FROM researcher ORDER BY `name` DESC;
show processlist;
- Time=34,已经执行了34s
恢复默认参数
# 检查默认值
show variables like '%quer%';
# 开启慢查询日志
set global slow_query_log=off;
# 设置慢查询阈值为1s
set global long_query_time=10;
# 检查设置值,若发现未生效,关闭当前会话(关闭数据库)重新打开,再检查
show variables like '%quer%';
# 重置表,包括自增ID
TRUNCATE TABLE researcher;