【问题标题】:MySQL query plan not using IndexMySQL 查询计划不使用索引
【发布时间】:2014-03-08 03:52:48
【问题描述】:

我有一个嵌套查询,我试图查看我的查询中是否有任何全表扫描。

explain delete from ACCESS where ACCESS.MESSAGEID in (select ID from MESSAGE where MESSAGE.CID = 'xzy67sd’)\G

子查询正在命中索引,但第一个查询没有使用索引。这是查询计划。

    *************************** 1. row ***************************
           id: 1
  select_type: PRIMARY
        table: ACCESS
         type: ALL
possible_keys: NULL
          key: NULL
      key_len: NULL
          ref: NULL
         rows: 18295
        Extra: Using where
*************************** 2. row ***************************
           id: 2
  select_type: DEPENDENT SUBQUERY
        table: MESSAGE
         type: unique_subquery
possible_keys: PRIMARY
          key: PRIMARY
      key_len: 8
          ref: func
         rows: 1
        Extra: Using where

但是,如果我将查询分开并检查查询计划,那么它正在使用索引。我无法理解为什么并寻找一些提示

explain delete from ACCESS where ACCESS.MESSAGEID in (2,3)\G
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: ACCESS
         type: range
possible_keys: ACCESS_ID1
key: ACCESS_ID1
      key_len: 8
          ref: const
         rows: 2
        Extra: Using where

子查询 select 语句返回常量,所以我输入整数而不是使用 select 语句,查询计划开始选择索引

select ID from MESSAGE where MESSAGE.CID = 'xzy67sd’)\G 

提前致谢

【问题讨论】:

    标签: mysql indexing


    【解决方案1】:

    在这里,您不需要子查询,而且作为一般规则,您不应该在 MySQL 中使用子查询,除非您确实需要。

    DELETE a
      FROM ACCESS a
      JOIN MESSAGE m ON m.ID = a.MESSAGEID
     WHERE m.CID = 'xzy67sd’;
    

    这将删除 ACCESS 中的行,同时保留 MESSAGE,因为在 DELETEFROM 之间仅列出了 ACCESS(通过其别名“a”),您可以在其中指定要从中删除匹配行的表.

    优化器应该适当地使用索引。

    https://dev.mysql.com/doc/refman/5.6/en/delete.html(多表语法)

    【讨论】:

      猜你喜欢
      • 2011-04-22
      • 2013-07-10
      • 2017-01-06
      • 2020-10-25
      • 1970-01-01
      • 2015-01-18
      • 2011-01-22
      • 1970-01-01
      相关资源
      最近更新 更多