@[toc]

General steps for tuning SQL statements

Show status commands

The show status command counts the execution frequency of SQL

Grammar:

-- Displays the values of all statistical parameters in the current session
show status like 'com_%';
Copy the code

The show status parameter is used to count the execution frequency of a specific SQL operation, for example, Com_XXX. XXX can be select, delete, update, insert, etc.

The show status command can also specify a storage engine, for example: Innodb_row_read Specifies the number of rows returned by the select query, Innodb_row_inserted specifies the number of rows inserted, Innodb_row_updated Specifies the number of rows updated by the UPDATE, Innodb_row_deleted Number of rows deleted by the DELETE command. Using these parameters, you can easily know whether the current database application is mainly used for insert updates or query operations, and the approximate proportion of various TYPES of SQL execution. The count of update operations is accumulated regardless of the number of commit and rollback times.

For transaction type applications, Com_commit and Com_rollback can be used to learn about transaction commit and rollback. For databases with frequent rollback operations, this may indicate problems in application writing.

The following parameters are used to understand the basic information about the database:

Connections: number of attempts to connect to the Mysql server. Uptime: server operating time Slow_queries: number of slow queries

Locate SQL statements that are executed inefficiently

  • –log-slow-queries[=file_name]. On startup, mysqld writes a log file containing all SQL statements whose execution time is longer than long_query_time seconds
  • Run the show processList command to check the current mysql thread, including the thread status, whether to shrink the table, and so on

Explain the execution plan for inefficient SQL

  • Select_type: specifies the type of SELECT, SIMPLE (without table joins or subqueries), PRIMARY PRIMARY query, UNION (the second or subsequent query of the UNION weight), SYBQUERY first SELECT of the subquery weight, etc.

  • Table: output result set table

  • MySQL > select * from ALL->index->range->eq_ref->const,system->NULL, from left to right

    • ALL: indicates full table scanning
    • Index: full index scan. MySQL traverses the entire index to search for matching rows
    • Range: index range scan, common <,<=,>,>=,between, etc
    • Ref: Returns a row of records matching a single value using a prefix scan with a non-unique index or unique index
    • Eq_ref: Similar to ref, except that the index used is unique
    • Const /system: There is at most one matching row in a single table, which is very fast to query
    • Select 1 from dual where 1 = 1; select * from dual where 1 = 1;
  • Possible_keys: indicates the possible_keys that can be queried

  • Key: indicates the actual index used in query

  • Key_len: the length used in the index field

  • “Rows” : indicates the number of scanned rows

  • Extra: A description and description of the performance, containing additional information that is important to the execution plan but not suitable for display in other columns.

Show Profiles and Show Profiles analyze SQL

MySQL supports show profiles and show profiles since 5.0.37. You can check whether MySQL supports profiles by running the select @@have_profiling command. Profiles are off by default and can be done through set profiling=1; Enabling session-level profiling allows you to better understand SQL execution through a profile. The usage of show profile is as follows

  1. Execute business SQL, such as select count(*) from payment;
  2. show profiles; This statement lists the SQL execution process in detail, including queryId, time, and SQL procedure
  3. show profile for query #ID; This statement details the show profiles; Each state and elapsed time of a thread in a queryId in
  4. MySQL supports all, CPU, block IO, Context swutch, page faults, and other itemizations to see which resources MySQL uses are consuming too much time. show profile cpu for query #ID

It can be seen that the time mainly occurs in Sending data and the time is mainly on CPU. The same SQL will not be so time-consuming under MyISAM engine, because MyISAM engine caches the number of rows of the table and returns them directly during statistics while InnoDB needs to scan the table.

mysql> select count(*) from payment;+ -- -- -- -- -- -- -- -- -- -- + | count (*) | + -- -- -- -- -- -- -- -- -- -- + | 16049 | + -- -- -- -- -- -- -- -- -- -- + 1 row in the set (0.00 SEC)
mysql> show profiles;+----------+------------+------------------------------+ | Query_ID | Duration | Query | + -- -- -- -- -- -- -- -- -- - + -- -- -- -- -- -- -- -- -- -- -- - + -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- + | 1 | | 0.00572250 select count (*) from payment | | | 2 0.00440850 | select count (*) from payment | | 3 | | 0.00435275 select count (*) from payment | + -- -- -- -- -- -- -- -- -- - + -- -- -- -- -- -- -- -- -- -- -- - + -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- + 3 rows in the set, 1 warning (0.00 SEC)
mysql> show profile for query 3;+ -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- - + -- -- -- -- -- -- -- -- -- -- + | Status | Duration | + -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- - + -- -- -- -- -- -- -- -- -- -- + | starting 0.000144 | | | 0.000012 checking permissions | | | Opening tables 0.000016 | | | init 0.000013 | | | System lock 0.000008 | | | Optimizing 0.000005 | | | statistics 0.000017 | | | preparing 0.000011 | | | executing 0.000002 | | | Sending data | 0.000013 0.004019 | | end | | | | | 0.000010 query end | closing tables 0.000006 | | | freeing the items 0.000053 | | | Cleaning up 0.000026 | | + -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- - + -- -- -- -- -- -- -- -- -- -- + 15 rows in the set, 1 warning (0.01 SEC)
mysql> show profile cpu for query 3;+----------------------+----------+----------+------------+ | Status | Duration | CPU_user | CPU_system | + -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- - + -- -- -- -- -- -- -- -- -- - + -- -- -- -- -- -- -- -- -- - + -- -- -- -- -- -- -- -- -- -- -- -- + | starting | 0.000144 0.000054 0.000029 | | | | checking Permissions | 0.000012 0.000006 0.000004 | | | | Opening tables | 0.000016 0.000011 0.000005 | | | | init | | 0.000013 0.000009 0.000005 | | | System lock | | | | 0.000003 0.000004 0.000008 | optimizing | | | | 0.000001 0.000003 0.000005 | The statistics | 0.000017 0.000011 0.000006 | | | | preparing | | | | 0.000004 0.000007 0.000011 | executing | | 0.000002 0.000001 0.000000 | | | Sending data | | | | 0.000000 0.004022 0.004019 0.000013 0.000010 | | | | end 0.000000 | | query End | 0.000010 0.000010 0.000000 | | | | closing tables | 0.000006 0.000006 0.000000 | | | | freeing the items | | 0.000053 0.000053 0.000000 | | | cleaning up | | | | 0.000000 0.000025 0.000026 + -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- - + -- -- -- -- -- -- -- -- -- - + -- -- -- -- -- -- -- -- -- - + -- -- -- -- -- -- -- -- -- -- -- -- + 15 rows in the set, 1 warning (0.01 SEC)Copy the code

How does the trace analysis optimizer select an execution plan

MySQL 5.6 provides A trace for SQL. The trace file can be used to understand why the optimizer chose plan A over plan B

Usage: First open trace, set the JSON format, and set the maximum memory size that trace can use to avoid incomplete display because the default memory is too small during parsing.

mysql> SET OPTIMIZER_TRACE="enabled=on",END_MARKERS_IN_JSON=on;
Query OK, 0 rows affected (0.00 sec)

mysql> SET OPTIMIZER_TRACE_MAX_MEM_SIZE=1000000;
Query OK, 0 rows affected (0.00 sec)
Copy the code

The service SQL is as follows:

mysql> select rental_id from rental where 1=1 and rental_date >= 'the 2005-05-25 04:00:00' and rental_date <= 'the 2005-05-25 05:00:00' and inventory_id=4466;+ -- -- -- -- -- -- -- -- -- -- -- + | rental_id | + -- -- -- -- -- -- -- -- -- -- - + 39 | | + -- -- -- -- -- -- -- -- -- -- -- + 1 row in the set (0.00 SEC)Copy the code

Finally, check information_schema.optimizer_trace to see how MySQL executes SQL:

mysql> SELECT * FROM INFORMATION_SCHEMA.OPTIMIZER_TRACE;
Copy the code

conclusion

After the above steps, the cause of the problem can be basically confirmed. At this time, appropriate measures can be taken to optimize and improve the execution efficiency.