当前位置:  开发笔记 > 编程语言 > 正文

MySQL视图是否总是进行全表扫描?

如何解决《MySQL视图是否总是进行全表扫描?》经验,为你挑选了1个好方法。

我正在尝试优化使用MySQL 5.1中的视图的查询.看来即使我从视图中选择1列,它也总是进行全表扫描.这是预期的行为吗?

对于我在下面第一个查询中指定的表,视图只是SELECT"这些表中的所有列 - NOT*".

这是我从组成视图的查询中选择索引列的PromotionID时的解释输出.正如您所看到的,它与视图上的输出有很大不同.

EXPLAIN SELECT pb.PromotionID FROM PromotionBase pb INNER JOIN PromotionCart pct ON pb.PromotionID = pct.PromotionID INNER JOIN PromotionCode pc ON pb.PromotionID = pc.PromotionID WHERE pc.PromotionCode = '5TAFF312C0NT'\G;
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: pc
         type: const
possible_keys: PRIMARY,fk_pc_pb
          key: PRIMARY
      key_len: 302
          ref: const
         rows: 1
        Extra:
*************************** 2. row ***************************
           id: 1
  select_type: SIMPLE
        table: pb
         type: const
possible_keys: PRIMARY
          key: PRIMARY
      key_len: 4
          ref: const
         rows: 1
        Extra: Using index
*************************** 3. row ***************************
           id: 1
  select_type: SIMPLE
        table: pct
         type: const
possible_keys: PRIMARY
          key: PRIMARY
      key_len: 4
          ref: const
         rows: 1
        Extra: Using index
3 rows in set (0.00 sec)

当我从视图中选择相同的东西时的输出

EXPLAIN SELECT vpc.PromotionID FROM vw_PromotionCode vpc  WHERE vpc.PromotionCode = '5TAFF312C0NT'\G;
*************************** 1. row ***************************
           id: 1
  select_type: PRIMARY
        table: 
         type: ALL
possible_keys: NULL
          key: NULL
      key_len: NULL
          ref: NULL
         rows: 5830
        Extra: Using where
*************************** 2. row ***************************
           id: 2
  select_type: DERIVED
        table: pcart
         type: index
possible_keys: PRIMARY
          key: PRIMARY
      key_len: 4
          ref: NULL
         rows: 33
        Extra: Using index
*************************** 3. row ***************************
           id: 2
  select_type: DERIVED
        table: pb
         type: eq_ref
possible_keys: PRIMARY
          key: PRIMARY
      key_len: 4
          ref: readyinteractive.pcart.PromotionID
         rows: 1
        Extra:
*************************** 4. row ***************************
           id: 2
  select_type: DERIVED
        table: pc
         type: ref
possible_keys: fk_pc_pb
          key: fk_pc_pb
      key_len: 4
          ref: readyinteractive.pb.PromotionID
         rows: 249
        Extra: Using where
*************************** 5. row ***************************
           id: 3
  select_type: UNION
        table: pp
         type: index
possible_keys: PRIMARY
          key: pp_p
      key_len: 4
          ref: NULL
         rows: 1
        Extra: Using index
*************************** 6. row ***************************
           id: 3
  select_type: UNION
        table: pb
         type: eq_ref
possible_keys: PRIMARY
          key: PRIMARY
      key_len: 4
          ref: readyinteractive.pp.PromotionID
         rows: 1
        Extra:
*************************** 7. row ***************************
           id: 3
  select_type: UNION
        table: pc
         type: ref
possible_keys: fk_pc_pb
          key: fk_pc_pb
      key_len: 4
          ref: readyinteractive.pb.PromotionID
         rows: 249
        Extra: Using where
*************************** 8. row ***************************
           id: 4
  select_type: UNION
        table: pcp
         type: index
possible_keys: PRIMARY
          key: pcp_cp
      key_len: 4
          ref: NULL
         rows: 1
        Extra: Using index
*************************** 9. row ***************************
           id: 4
  select_type: UNION
        table: pb
         type: eq_ref
possible_keys: PRIMARY
          key: PRIMARY
      key_len: 4
          ref: readyinteractive.pcp.PromotionID
         rows: 1
        Extra:
*************************** 10. row ***************************
           id: 4
  select_type: UNION
        table: pc
         type: ref
possible_keys: fk_pc_pb
          key: fk_pc_pb
      key_len: 4
          ref: readyinteractive.pb.PromotionID
         rows: 249
        Extra: Using where
*************************** 11. row ***************************
           id: 5
  select_type: UNION
        table: ppc
         type: index
possible_keys: PRIMARY
          key: ppc_pc
      key_len: 4
          ref: NULL
         rows: 1
        Extra: Using index
*************************** 12. row ***************************
           id: 5
  select_type: UNION
        table: pb
         type: eq_ref
possible_keys: PRIMARY
          key: PRIMARY
      key_len: 4
          ref: readyinteractive.ppc.PromotionID
         rows: 1
        Extra:
*************************** 13. row ***************************
           id: 5
  select_type: UNION
        table: pc
         type: ref
possible_keys: fk_pc_pb
          key: fk_pc_pb
      key_len: 4
          ref: readyinteractive.pb.PromotionID
         rows: 249
        Extra: Using where
*************************** 14. row ***************************
           id: 6
  select_type: UNION
        table: ppt
         type: index
possible_keys: PRIMARY
          key: ppt_pt
      key_len: 4
          ref: NULL
         rows: 1
        Extra: Using index
*************************** 15. row ***************************
           id: 6
  select_type: UNION
        table: pb
         type: eq_ref
possible_keys: PRIMARY
          key: PRIMARY
      key_len: 4
          ref: readyinteractive.ppt.PromotionID
         rows: 1
        Extra:
*************************** 16. row ***************************
           id: 6
  select_type: UNION
        table: pc
         type: ref
possible_keys: fk_pc_pb
          key: fk_pc_pb
      key_len: 4
          ref: readyinteractive.pb.PromotionID
         rows: 249
        Extra: Using where
*************************** 17. row ***************************
           id: NULL
  select_type: UNION RESULT
        table: 
         type: ALL
possible_keys: NULL
          key: NULL
      key_len: NULL
          ref: NULL
         rows: NULL
        Extra:
17 rows in set (0.18 sec)

Noah Goodric.. 9

MySQL中的视图没有编入索引,因此它们的本质需要在每次访问时进行完整扫描.一般来说,这使得Views真的只对你有一个相当复杂的静态查询返回一个小结果集并且你打算每次都抓住整个结果集的情况有用.

编辑:当然,视图将使用基础表上的索引,以便优化View本身(否则它们根本就没有任何意义)但由于View上没有索引,因此不可能使用WHERE查询要优化的视图.

构建视图的索引无论如何都是昂贵的,因为虽然我没有尝试分析任何视图,但我相当确定在后台构造临时表,然后返回结果集.它已经花了很多时间来构造临时表,我不想要一个也试图猜测需要哪些索引的视图.这引出了第二点,即MySQL目前没有提供一种方法来指定用于View的索引,那么它如何知道哪些字段需要被索引?它会根据您的查询猜测吗?

您可以考虑使用临时表,因为您可以在临时表中的字段上指定索引.但是,从经验来看,这往往非常非常缓慢.

如果所有这个视图都包含SELECT ALL FROM table1,table2,table3; 那么我不得不问为什么这个查询需要在View中呢?如果由于某种原因它绝对必要,您可能希望使用存储过程来封装查询,因为您将能够获得优化的性能,同时保持对结果集的数据库的更简单调用的好处.



1> Noah Goodric..:

MySQL中的视图没有编入索引,因此它们的本质需要在每次访问时进行完整扫描.一般来说,这使得Views真的只对你有一个相当复杂的静态查询返回一个小结果集并且你打算每次都抓住整个结果集的情况有用.

编辑:当然,视图将使用基础表上的索引,以便优化View本身(否则它们根本就没有任何意义)但由于View上没有索引,因此不可能使用WHERE查询要优化的视图.

构建视图的索引无论如何都是昂贵的,因为虽然我没有尝试分析任何视图,但我相当确定在后台构造临时表,然后返回结果集.它已经花了很多时间来构造临时表,我不想要一个也试图猜测需要哪些索引的视图.这引出了第二点,即MySQL目前没有提供一种方法来指定用于View的索引,那么它如何知道哪些字段需要被索引?它会根据您的查询猜测吗?

您可以考虑使用临时表,因为您可以在临时表中的字段上指定索引.但是,从经验来看,这往往非常非常缓慢.

如果所有这个视图都包含SELECT ALL FROM table1,table2,table3; 那么我不得不问为什么这个查询需要在View中呢?如果由于某种原因它绝对必要,您可能希望使用存储过程来封装查询,因为您将能够获得优化的性能,同时保持对结果集的数据库的更简单调用的好处.

推荐阅读
jerry613
这个屌丝很懒,什么也没留下!
DevBox开发工具箱 | 专业的在线开发工具网站    京公网安备 11010802040832号  |  京ICP备19059560号-6
Copyright © 1998 - 2020 DevBox.CN. All Rights Reserved devBox.cn 开发工具箱 版权所有