MySQL SQL分析

Jun 01, 2016 pm 01:28 PM

bitsCN.com

MySQL SQL分析 - 参数化查询vs query cache功能

 

query cache,  mysql 5 开始附带的一个功能, 与引擎无关, 只与数据查询语法相关。

 

测试描述: 当前使用中是 MySQL-5.6.14 Linux RHEL6  64 位系统产生环境,  使用 INNODB 引擎, 分配 innodb 2g 内存空间

 

[root@TiYanPlat ~]# uname -aLinux TiYanPlat 2.6.32-358.el6.x86_64 #1 SMP Tue Jan 29 11:47:41 EST 2013 x86_64 x86_64x86_64 GNU/Linuxmysql> select version();+-----------+| version() |+-----------+| 5.6.14    |+-----------+1 row in set (0.00 sec)mysql> show variables like 'innodb_buffer_pool_size';+-------------------------+------------+| Variable_name           | Value      |+-------------------------+------------+| innodb_buffer_pool_size | 2147483648 |+-------------------------+------------+1 row in set (0.01 sec)
로그인 후 복사

Query cache 功能:

利用 qeury_cache_size 定义内存大小, 内存用于把用户 SQL 放入内存中, 包括 SQL 语句, 包括SQL 语句执行的结果

假如下一次查询时使用相同的 SQL 语句, 则直接从内存中获得结果, 不再进行 SQL 分析, 不在进行磁盘 I/O 读数据。加速数据查询返回结果。

实现目标,开启 QCACHE 功能, 如 my.cnf 定义

query-cache-size=16777216

query-cache-type=ON

查询数据库中是否使用当前功能

mysql> show status like '%qcache%';+-------------------------+----------+| Variable_name           | Value    |+-------------------------+----------+| Qcache_free_blocks      | 440      || Qcache_free_memory      | 12306960 || Qcache_hits             | 13176    || Qcache_inserts          | 29777    || Qcache_lowmem_prunes    | 0        || Qcache_not_cached       | 45862    || Qcache_queries_in_cache | 2098     || Qcache_total_blocks     | 4701     |+-------------------------+----------+8 rows in set (0.02 sec)
로그인 후 복사

参数返回结果不再一一详细描述, 自行参考官方文档, 从上返回结果可以看到,使用中的数据库 SQL 命中率 (Qcache_hits) 并不理想,原因与业务有关。

SQL 分析一, 使用了 QUERY CACHE 的好处

原理, 利用 EXPLAIN 分析当前 SQL 执行计划, 利用 PROFILE 功能分析当前 SQL 执行计划,过程

执行下面语句进行分析

mysql> explain select tbcrbtnumb0_.id as id41_, tbcrbtnumb0_.business_ring_id as business2_41_, tbcrbtnumb0_.application_no as applicat3_41_, tbcrbtnumb0_.mobile as mobile41_ from tb_crbt_numbers_load tbcrbtnumb0_ where 1=1 and tbcrbtnumb0_.business_ring_id=11024;+----+-------------+--------------+------+---------------+------+---------+------+--------+-------------+| id | select_type | table        | type | possible_keys | key  | key_len | ref  | rows   | Extra       |+----+-------------+--------------+------+---------------+------+---------+------+--------+-------------+|  1 | SIMPLE      | tbcrbtnumb0_ | ALL  | NULL          | NULL | NULL    | NULL |180182 | Using where |+----+-------------+--------------+------+---------------+------+---------+------+--------+-------------+1 row in set (0.00 sec)当前语法执行的是全表扫描,另外,需要从 180182 行中扫描相关结果SQL 分析二, 判断第一次执行该 SQL 时候的执行过程mysql> set profiling=1;mysql> select tbcrbtnumb0_.id as id41_, tbcrbtnumb0_.business_ring_id as business2_41_, tbcrbtnumb0_.application_no as applicat3_41_, tbcrbtnumb0_.mobile as mobile41_ from tb_crbt_numbers_load tbcrbtnumb0_ where 1=1 and tbcrbtnumb0_.business_ring_id=11024;+-------+---------------+---------------+-------------+| id41_ | business2_41_ | applicat3_41_ | mobile41_   |+-------+---------------+---------------+-------------+| 30838 |         11024 | TH20121127229 | 02038688592 |+-------+---------------+---------------+-------------+1 row in set (0.30 sec)mysql> show profile;                                                                                                          +--------------------------------+----------+| Status                         | Duration |+--------------------------------+----------+| starting                       | 0.000191 || Waiting for query cache lock   | 0.000023 || init                           | 0.000090 || checking query cache for query | 0.000499 || checking permissions           | 0.000030 || Opening tables                 | 0.000131 || init                           | 0.000277 || System lock                    | 0.000042 || Waiting for query cache lock   | 0.000005 || System lock                    | 0.000443 || optimizing                     | 0.000364 || statistics                     | 0.000107 || preparing                      | 0.000059 || executing                      | 0.000019 || Sending data                   | 0.290067 || end                            | 0.000483 || query end                      | 0.000169 || closing tables                 | 0.000158 || freeing items                  | 0.000252 || Waiting for query cache lock   | 0.000063 || freeing items                  | 0.000305 || Waiting for query cache lock   | 0.000015 || freeing items                  | 0.000095 || storing result in query cache  | 0.000145 || cleaning up                    | 0.000330 |+--------------------------------+----------+25 rows in set, 1 warning (0.01 sec)
로그인 후 복사

从上面看出, 第一次执行该 SQL, MySQL 需要对 SQL 进行锁缓存,初始化,从缓存中查询是否具备之前缓存过的 SQL,检查用户权限, 表权限,打开表,锁定内存,定制执行计划,执行语句,把数据从磁盘中放入内存中操作,关闭表,锁定数据, 缓存数据等操作, 工作原理与 ORACLE 类似

按照 QUERY CACHE 原则, 假如 SQL 语句改变 (tbcrbtnumb0_.business_ring_id=11024) 替换该变量值, 那么该 SQL 会被看作为一个新的 SQL, 这个时候, MySQL 将会对整个 SQL 做一次全新的操作, 如上(黄线标注描述)

分析 SQL 三

mysql> select tbcrbtnumb0_.id as id41_, tbcrbtnumb0_.business_ring_id as business2_41_, tbcrbtnumb0_.application_no as applicat3_41_, tbcrbtnumb0_.mobile as mobile41_ from tb_crbt_numbers_load tbcrbtnumb0_ where 1=1 and tbcrbtnumb0_.business_ring_id=11021;+-------+---------------+---------------+-------------+| id41_ | business2_41_ | applicat3_41_ | mobile41_   |+-------+---------------+---------------+-------------+| 30835 |         11021 | TH20121127259 | 02038688592 |+-------+---------------+---------------+-------------+1 row in set (0.34 sec)mysql> show profile;+--------------------------------+----------+| Status                         | Duration |+--------------------------------+----------+| starting                       | 0.000795 || Waiting for query cache lock   | 0.000077 || init                           | 0.000045 || checking query cache for query | 0.000337 || checking permissions           | 0.000040 || Opening tables                 | 0.000113 || init                           | 0.000488 || System lock                    | 0.000050 || Waiting for query cache lock   | 0.000030 || System lock                    | 0.000289 || optimizing                     | 0.000512 || statistics                     | 0.000278 || preparing                      | 0.000078 || executing                      | 0.000028 || Sending data                   | 0.322662 || end                            | 0.004777 || query end                      | 0.001703 || closing tables                 | 0.000526 || freeing items                  | 0.000874 || Waiting for query cache lock   | 0.000311 || freeing items                  | 0.001809 || Waiting for query cache lock   | 0.000105 || freeing items                  | 0.000184 || storing result in query cache  | 0.000966 || cleaning up                    | 0.000678 |+--------------------------------+----------+25 rows in set, 1 warning (0.00 sec)
로그인 후 복사

上 SQL二,三结果可以看到, 当 WHERE 条件改变, MySQL 会把这两个 SQL 识别为一个新的 SQL, 需要重新操作。

SQL 分析四, 假如我们重新执行 SQL 三操作,看看结果如何?(注意,这个时候 QUERY CACHE 真正发挥作用)

mysql> select tbcrbtnumb0_.id as id41_, tbcrbtnumb0_.business_ring_id as business2_41_, tbcrbtnumb0_.application_no as applicat3_41_, tbcrbtnumb0_.mobile as mobile41_ from tb_crbt_numbers_load tbcrbtnumb0_ where 1=1 and tbcrbtnumb0_.business_ring_id=11021;+-------+---------------+---------------+-------------+| id41_ | business2_41_ | applicat3_41_ | mobile41_   |+-------+---------------+---------------+-------------+| 30835 |         11021 | TH20121127259 | 02038688592 |+-------+---------------+---------------+-------------+1 row in set (0.00 sec)mysql> show profile;                                                                                                          +--------------------------------+----------+| Status                         | Duration |+--------------------------------+----------+| starting                       | 0.001367 || Waiting for query cache lock   | 0.000071 || init                           | 0.000027 || checking query cache for query | 0.000163 || checking privileges on cached  | 0.000129 || checking permissions           | 0.000386 || sending cached result to clien | 0.000164|| cleaning up                    | 0.000079 |+--------------------------------+----------+8 rows in set, 1 warning (0.01 sec)
로그인 후 복사

参考 SQL 分析三,四,数据查询需要使用的时间(绿色标注部分)很明显, SQL 分析四返回速度块了很多,另外,从系统返回的 SQL 分析看出来,系统直接从缓存中返回数据给客户, 没有重复进行 SQL 分析及磁盘 I/O 操作。(蓝色标注部分) 因此, QEURY CACHE 明显加速了 SQL 返回结果。

但必须注意,只有两个 SQL 相同的情况下,才能够获得 QUERY CACHE 的优点。

参数化查询

参数化查询能够在一定情况下避免了 SQL 注入, 而 ORACLE 也比较推荐使用参数化查询, ORACLE 每次执行 SQL (无论 SQL 是否语法一致)都存在 SQL 分析,

假如SQL语法不一样,则进行硬解析,需要重新定制执行计划

假如SQL语法不一致则进行软解析,避免重复定制执行计划,减少 CPU 消耗,增加 SQL 语句返回时间。

MySQL 官方文档中并没有提出到这点。

对 MySQL 进行参数化查询分析

SQL 参数化分析一

mysql> set @num=11204;mysql> select tbcrbtnumb0_.id as id41_, tbcrbtnumb0_.business_ring_id as business2_41_, tbcrbtnumb0_.application_no as applicat3_41_, tbcrbtnumb0_.mobile as mobile41_ from tb_crbt_numbers_load tbcrbtnumb0_ where 1=1 and tbcrbtnumb0_.business_ring_id=@num;+-------+---------------+---------------+-----------+| id41_ | business2_41_ | applicat3_41_ | mobile41_ |+-------+---------------+---------------+-----------+| 31051 |         11204 | 1222570       | 85237810  || 31052 |         11204 | 1222570       | 82685386  || 31053 |         11204 | 1222570       | 82783689  || 31054 |         11204 | 1222570       | 82685106  || 31055 |         11204 | 1222570       | 38880051  |+-------+---------------+---------------+-----------+5 rows in set (0.37 sec)mysql> show profile;                                                                                                          +--------------------------------+----------+| Status                         | Duration |+--------------------------------+----------+| starting                       | 0.001858 || Waiting for query cache lock   | 0.000089 || init                           | 0.000150 || checking query cache for query | 0.001143 || checking permissions           | 0.000970 || Opening tables                 | 0.000544 || init                           | 0.000743 || System lock                    | 0.000170 || optimizing                     | 0.000332 || statistics                     | 0.000293 || preparing                      | 0.000134 || executing                      | 0.000057 || Sending data                   | 0.438626 || end                            | 0.000694 || query end                      | 0.000221 || closing tables                 | 0.000300 || freeing items                  | 0.000521 || cleaning up                    | 0.000360 |+--------------------------------+----------+18 rows in set, 1 warning (0.01 sec)
로그인 후 복사

变量值不变情况下,重复执行该 SQL 语句

mysql> select tbcrbtnumb0_.id as id41_, tbcrbtnumb0_.business_ring_id as business2_41_, tbcrbtnumb0_.application_no as applicat3_41_, tbcrbtnumb0_.mobile as mobile41_ from tb_crbt_numbers_load tbcrbtnumb0_ where 1=1 and tbcrbtnumb0_.business_ring_id=@num;+-------+---------------+---------------+-----------+| id41_ | business2_41_ | applicat3_41_ | mobile41_ |+-------+---------------+---------------+-----------+| 31051 |         11204 | 1222570       | 85237810  || 31052 |         11204 | 1222570       | 82685386  || 31053 |         11204 | 1222570       | 82783689  || 31054 |         11204 | 1222570       | 82685106  || 31055 |         11204 | 1222570       | 38880051  |+-------+---------------+---------------+-----------+5 rows in set (0.34 sec)mysql> show profile;                                                                                                          +--------------------------------+----------+| Status                         | Duration |+--------------------------------+----------+| starting                       | 0.001188 || Waiting for query cache lock   | 0.000052 || init                           | 0.000030 || checking query cache for query | 0.001791 || checking permissions           | 0.000172 || Opening tables                 | 0.000614 || init                           | 0.001346 || System lock                    | 0.000268 || optimizing                     | 0.000626 || statistics                     | 0.000674 || preparing                      | 0.000439 || executing                      | 0.000028 || Sending data                   | 0.327278 || end                            | 0.000649 || query end                      | 0.000217 || closing tables                 | 0.000408 || freeing items                  | 0.000692 || cleaning up                    | 0.000570 |+--------------------------------+----------+18 rows in set, 1 warning (0.01 sec)
로그인 후 복사

 

分析结果

从数据返回时间上看来(蓝色标注), 使用参数化查询,并没有在时间返回上获得优势。

从SQL执行计划上看来(紫色标注), 相同 SQL 语句使用参数化查询,系统同样会重新定制执行计划,产生磁盘I/O, 并没有想 ORACLE 一样获得性能上的优化。

另外,参数化查询是不会在 query cache 内存块中取结果的。

 

可以看做每次使用参数化查询都会被认为是一个全新的 SQL 进行分析, (没有学习到 ORACLE 的精髓,比较失望)

 

注意,使用参数化查询时, 即时使用 SQL_cache  语法, 也无法使用 query cache 功能。

 

常见开发下有几种选择

1. 直接把 SQL 变量值提交至 MySQL API 执行,(可利用 QUERY CACHE 功能,有部分 SQL 能够进行数据加速)但可能会遇到 SQL 注入, 升级程序麻烦。

2. 利用  procudure, function 等功能先吧 SQL 进行打包然后再调用执行, 类似参数化查询, 无法获得 QUERY CACHE 功能, 令程序清晰, 程序升级,修改比较方便, 并且有效防止了 SQL 注入。

 

bitsCN.com
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

Video Face Swap

Video Face Swap

완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

인기 기사

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

MySQL의 역할 : 웹 응용 프로그램의 데이터베이스 MySQL의 역할 : 웹 응용 프로그램의 데이터베이스 Apr 17, 2025 am 12:23 AM

웹 응용 프로그램에서 MySQL의 주요 역할은 데이터를 저장하고 관리하는 것입니다. 1. MySQL은 사용자 정보, 제품 카탈로그, 트랜잭션 레코드 및 기타 데이터를 효율적으로 처리합니다. 2. SQL 쿼리를 통해 개발자는 데이터베이스에서 정보를 추출하여 동적 컨텐츠를 생성 할 수 있습니다. 3.mysql은 클라이언트-서버 모델을 기반으로 작동하여 허용 가능한 쿼리 속도를 보장합니다.

InnoDB Redo Logs 및 Undo Logs의 역할을 설명하십시오. InnoDB Redo Logs 및 Undo Logs의 역할을 설명하십시오. Apr 15, 2025 am 12:16 AM

InnoDB는 Redologs 및 Undologs를 사용하여 데이터 일관성과 신뢰성을 보장합니다. 1. Redologs는 사고 복구 및 거래 지속성을 보장하기 위해 데이터 페이지 수정을 기록합니다. 2. 결점은 원래 데이터 값을 기록하고 트랜잭션 롤백 및 MVCC를 지원합니다.

MySQL : 세계에서 가장 인기있는 데이터베이스 소개 MySQL : 세계에서 가장 인기있는 데이터베이스 소개 Apr 12, 2025 am 12:18 AM

MySQL은 오픈 소스 관계형 데이터베이스 관리 시스템으로, 주로 데이터를 신속하고 안정적으로 저장하고 검색하는 데 사용됩니다. 작업 원칙에는 클라이언트 요청, 쿼리 해상도, 쿼리 실행 및 반환 결과가 포함됩니다. 사용의 예로는 테이블 작성, 데이터 삽입 및 쿼리 및 조인 작업과 같은 고급 기능이 포함됩니다. 일반적인 오류에는 SQL 구문, 데이터 유형 및 권한이 포함되며 최적화 제안에는 인덱스 사용, 최적화 된 쿼리 및 테이블 분할이 포함됩니다.

MySQL의 장소 : 데이터베이스 및 프로그래밍 MySQL의 장소 : 데이터베이스 및 프로그래밍 Apr 13, 2025 am 12:18 AM

데이터베이스 및 프로그래밍에서 MySQL의 위치는 매우 중요합니다. 다양한 응용 프로그램 시나리오에서 널리 사용되는 오픈 소스 관계형 데이터베이스 관리 시스템입니다. 1) MySQL은 웹, 모바일 및 엔터프라이즈 레벨 시스템을 지원하는 효율적인 데이터 저장, 조직 및 검색 기능을 제공합니다. 2) 클라이언트 서버 아키텍처를 사용하고 여러 스토리지 엔진 및 인덱스 최적화를 지원합니다. 3) 기본 사용에는 테이블 작성 및 데이터 삽입이 포함되며 고급 사용에는 다중 테이블 조인 및 복잡한 쿼리가 포함됩니다. 4) SQL 구문 오류 및 성능 문제와 같은 자주 묻는 질문은 설명 명령 및 느린 쿼리 로그를 통해 디버깅 할 수 있습니다. 5) 성능 최적화 방법에는 인덱스의 합리적인 사용, 최적화 된 쿼리 및 캐시 사용이 포함됩니다. 모범 사례에는 거래 사용 및 준비된 체계가 포함됩니다

MySQL을 사용하는 이유는 무엇입니까? 혜택과 장점 MySQL을 사용하는 이유는 무엇입니까? 혜택과 장점 Apr 12, 2025 am 12:17 AM

MySQL은 성능, 신뢰성, 사용 편의성 및 커뮤니티 지원을 위해 선택됩니다. 1.MYSQL은 효율적인 데이터 저장 및 검색 기능을 제공하여 여러 데이터 유형 및 고급 쿼리 작업을 지원합니다. 2. 고객-서버 아키텍처 및 다중 스토리지 엔진을 채택하여 트랜잭션 및 쿼리 최적화를 지원합니다. 3. 사용하기 쉽고 다양한 운영 체제 및 프로그래밍 언어를 지원합니다. 4. 강력한 지역 사회 지원을 받고 풍부한 자원과 솔루션을 제공합니다.

MySQL 대 기타 프로그래밍 언어 : 비교 MySQL 대 기타 프로그래밍 언어 : 비교 Apr 19, 2025 am 12:22 AM

다른 프로그래밍 언어와 비교할 때 MySQL은 주로 데이터를 저장하고 관리하는 데 사용되는 반면 Python, Java 및 C와 같은 다른 언어는 논리적 처리 및 응용 프로그램 개발에 사용됩니다. MySQL은 데이터 관리 요구에 적합한 고성능, 확장 성 및 크로스 플랫폼 지원으로 유명하며 다른 언어는 데이터 분석, 엔터프라이즈 애플리케이션 및 시스템 프로그래밍과 같은 해당 분야에서 이점이 있습니다.

MySQL Index Cardinality는 쿼리 성능에 어떤 영향을 미칩니 까? MySQL Index Cardinality는 쿼리 성능에 어떤 영향을 미칩니 까? Apr 14, 2025 am 12:18 AM

MySQL Index Cardinality는 쿼리 성능에 중대한 영향을 미칩니다. 1. 높은 카디널리티 인덱스는 데이터 범위를보다 효과적으로 좁히고 쿼리 효율성을 향상시킬 수 있습니다. 2. 낮은 카디널리티 인덱스는 전체 테이블 스캔으로 이어질 수 있으며 쿼리 성능을 줄일 수 있습니다. 3. 관절 지수에서는 쿼리를 최적화하기 위해 높은 카디널리티 시퀀스를 앞에 놓아야합니다.

MySQL : 소기업에서 대기업에 이르기까지 MySQL : 소기업에서 대기업에 이르기까지 Apr 13, 2025 am 12:17 AM

MySQL은 소규모 및 대기업에 적합합니다. 1) 소기업은 고객 정보 저장과 같은 기본 데이터 관리에 MySQL을 사용할 수 있습니다. 2) 대기업은 MySQL을 사용하여 대규모 데이터 및 복잡한 비즈니스 로직을 처리하여 쿼리 성능 및 트랜잭션 처리를 최적화 할 수 있습니다.

See all articles