数据库
首页 > 数据库> > mysql – 使用大量内部联接来改进查询到wp_postmeta,一个键/值表

mysql – 使用大量内部联接来改进查询到wp_postmeta,一个键/值表

作者:互联网

我正在使用正在执行以下查询的wordpress网站,但我看到这个查询正在进行许多内部联接,并且网站需要很长时间才能加载并且下载很多,我一直在尝试创建一个产生相同结果的查询但还没有成功

我想知道什么是更好的方法来做到这一点

SELECT *
FROM wp_posts
INNER JOIN wp_postmeta color ON wp_posts.ID = color.post_id 
INNER JOIN wp_postmeta transmission ON wp_posts.ID = transmission.post_id 
INNER JOIN wp_postmeta model ON wp_posts.ID = model.post_id 
INNER JOIN wp_postmeta brand ON wp_posts.ID = brand.post_id 

AND color.meta_key = 'color' 
AND color.meta_value = 'red' 
AND transmission.meta_key = 'transmission' 
AND transmission.meta_value = 'auto' 
AND model.meta_key = 'model' 
AND model.meta_value = 'model' 
AND brand.meta_key = 'brand' 
AND brand.meta_value = 'brand'

AND wp_posts.post_status = 'publish'
AND wp_posts.post_type = 'car'
ORDER BY wp_posts.post_title

这是解释输出.

+----+-------------+-----------+--------+-----------------------------+----------+---------+------------------------+------+----------------------------------------------+
| id | select_type | table         | type   | possible_keys               | key      | key_len | ref                          | rows | Extra                                        |
+----+-------------+-----------+--------+-----------------------------+----------+---------+------------------------+------+----------------------------------------------+
|  1 | SIMPLE      | color         | ref    | post_id,meta_key            | meta_key | 768     | const                        |  629 | Using where; Using temporary; Using filesort |
|  1 | SIMPLE      | wp_posts      | eq_ref | PRIMARY,type_status_date,ID | PRIMARY  | 8       | tmcdb.color.post_id          |    1 | Using where                                  |
|  1 | SIMPLE      | brand         | ref    | post_id,meta_key            | post_id  | 8       | tmcdb.wp_posts.ID            |    4 | Using where                                  |
|  1 | SIMPLE      | transmission  | ref    | post_id,meta_key            | post_id  | 8       | tmcdb.color.post_id          |    4 | Using where                                  |
|  1 | SIMPLE      | model         | ref    | post_id,meta_key            | post_id  | 8       | tmcdb.transmission.post_id   |    4 | Using where                                  |
+----+-------------+-----------+--------+-----------------------------+----------+---------+------------------------+------+----------------------------------------------+

WordPress schema here.

解决方法:

您似乎正在尝试获取每个类型为car的帖子一行的结果集.看起来你想要在帖子中显示每辆车的各种属性,并且这些属性都隐藏在postmeta中.

专业提示:除非您完全知道为什么要这样做,否则切勿在软件中使用SELECT *.特别是对于包含大量JOIN操作的查询,SELECT *返回许多无意义和冗余的列.

有一个查询设计技巧要知道WordPress postmeta表.如果要获取特定属性,请执行以下操作:

 SELECT p.ID, p.post_title,
        color.meta_value AS color
   FROM wp_posts AS p
   LEFT JOIN wp_postmeta AS color ON p.ID = color.post_id AND 'color' = color.meta_key
  WHERE p.post_status = 'publish'
    AND /* etc etc */

在做你想做的事情时理解这种模式是非常重要的.这种模式是必需的,因为postmeta是一种特殊类型的表,称为商店.这里发生了什么?一些东西:

>使用此模式,每个帖子都会有一行,其中包含posts表中的一些列和postmeta表中的特定属性.
>您左键联接postmeta表,因此如果缺少该属性,您仍会获得一行.
>您正在使用postmeta表的别名.这是postmeta AS的颜色.
>您在连接的ON条件中包含meta_key的选择器(此处为’color’= color.meta_key).
>您在SELECT子句中使用别名来显示具有适当列名的postmeta.meta_value项.这是color.meta_value AS颜色.

一旦习惯了这种模式,就可以通过级联LEFT JOIN操作将其叠加起来,以获得许多不同的属性,就像这样.

     SELECT wp_posts.ID, wp_posts.post_title, wp_posts.whatever,
            color.meta_value        AS color,
            transmission.meta_value AS transmission,
            model.meta_value        AS model,
            brand.meta_value        AS brand
       FROM wp_posts

  LEFT JOIN wp_postmeta  AS color 
         ON wp_posts.ID = color.post_id        AND color.meta_key='color'

  LEFT JOIN wp_postmeta  AS transmission
         ON wp_posts.ID = transmission.post_id AND transmission.meta_key='transmission'

  LEFT JOIN wp_postmeta  AS model
         ON wp_posts.ID = model.post_id        AND model.meta_key='model'

  LEFT JOIN wp_postmeta  AS  brand
         ON wp_posts.ID = brand.post_id        AND brand.meta_key='brand'

      WHERE wp_posts.post_status = 'publish'
        AND wp_posts.post_type = 'car'
   ORDER BY wp_posts.post_title

我在这个查询上做了一堆缩进,以便更容易看到模式.您可能更喜欢不同的缩进样式.

很难知道为什么在您的问题中出现查询性能问题.这可能是因为您在所有INNER JOIN操作中获得了组合爆炸,然后进行了过滤.但无论如何,你展示的查询可能没有返回任何行.

如果您仍然遇到性能问题,请尝试在(post_id,meta_key,meta_value)列上的postmeta上创建复合索引.如果你正在创建一个WordPress插件,这可能是插件安装时的一项工作.

标签:entity-attribute-value,sql,mysql,wordpress,database,key-value,entity-attribute-v
来源: https://codeday.me/bug/20190926/1822235.html