Перед синтаксическим анализом запросы сравниваются, поэтому запросы
SELECT * FROM tbl_name
и
Select * from tbl_name
для кэша запросов рассматриваются как различные, поскольку они должны быть абсолютно одинаковыми (байт в байт), чтобы рассматриваться как идентичные. Помимо этого, запрос может трактоваться как отличающийся, если, например, какой-либо клиент использует протокол соединения нового формата или иной набор символов, чем другой клиент.
Запросы, использующие различные базы данных, различные версии протоколов или различные наборы символов по умолчанию, рассматриваются как различные и кэшируются раздельно.
Рассматриваемый кэш надежно работает для запросов вида SELECT CALC_ROWS ...
и SELECT FOUND_ROWS() ...
, так как число найденных строк всегда
хранится в кэше.
Если результат запроса вернулся из кеша запросов, тогда статусная переменная
Com_select
не будет увеличена, но вместо нее будет увеличена Qcache_hits
.
See Раздел 6.9.4, «Статус и поддержка кэша запросов».
При изменениях таблицы (INSERT
, UPDATE
, DELETE
,
TRUNCATE
, ALTER
или DROP TABLE|DATABASE
), все кэшированные
запросы, использовавшие данную таблицу (возможно, через таблицу
MRG_MyISAM
!), становятся недействительными и удаляются из кэша.
Если изменения были произведены в поддерживающих транзакции таблицах вида
InnoDB
, то все кэшированные запросы становятся недействительными при
выполнении команды COMMIT
.
Запрос не будет кэширован, если содержит одну из приведенных ниже функций:
Функция | Функция | Функция |
Определяемые пользователем функции (UDF)
|
CONNECTION_ID
|
FOUND_ROWS
|
GET_LOCK
|
RELEASE_LOCK
|
LOAD_FILE
|
MASTER_POS_WAIT
|
NOW
|
SYSDATE
|
CURRENT_TIMESTAMP
|
CURDATE
|
CURRENT_DATE
|
CURTIME
|
CURRENT_TIME
|
DATABASE
|
ENCRYPT (с одним параметром)
|
LAST_INSERT_ID
|
RAND
|
UNIX_TIMESTAMP (без параметров)
|
USER
|
BENCHMARK
|
Запрос также не будет кэширован, если он содержит переменные пользователя,
работает с системными таблицами mysql
,
или выражен в форме SELECT ... IN SHARE MODE
,
SELECT ... INTO OUTFILE ...
,
SELECT ... INTO DUMPFILE ...
или в форме SELECT * FROM AUTOINCREMENT_FIELD IS NULL
(для получения последнего ID - это для ODBC).
Однако FOUND ROWS()
возвратит правильную величину, даже если из кэша был
выбран предыдущий запрос.
В случае если запрос не использует таблиц, или использует временные таблицы, или если пользователь обладает привилегиями уровня столбца на какую-либо из задействованных таблиц, запрос не будет кеширован.
Перед выборкой запроса из кэша запросов MySQL проверит, обладает ли
пользователь привилегией SELECT
для всех включенных баз данных и таблиц.
Если это не так, то результат кэширования не используется.