MySQL-添加多个派生表时的慢查询-优化

时间:2018-07-10 15:15:42

标签: mysql optimization

对于我的查询,底部的两个派生表正在导致此查询的疯狂减速。查询按原样执行大约需要45-55秒。现在,当我仅删除那些派生表之一(与哪一个无关)时,查询时间将降至0.1-0.3秒。我的问题;拥有多个派生表是否存在问题?有没有更好的方法来执行此操作?我的索引似乎都是正确的,我还将包括此查询中的解释。

select t.name as team, u.name as "REP NAME", 
  count(distinct activity.id) as "TOTAL VISITS", 
  count(distinct activity.account_id) as "UNIQUE VISITS",
  count(distinct placement.id) as "COMMITMENTS ADDED",

  CASE WHEN 
    count(distinct activity.account_id) = 0 THEN (count(distinct 
    placement.id) / 1) 
    else (cast(count(distinct placement.id) as decimal(10,2)) / 
    cast(count(distinct activity.account_id) as decimal(10,2)))
  end as "UNIQUE VISIT TO COMMITMENT %",

  case when o.mode='basic' then count(distinct placement.id) else 
    count(distinct(case when placement.commitmentstatus='fullfilled' 
    then placement.id else 0 end)) 
  end as "COMMITMENTS FULFILLED",

  case when o.mode='basic' then 1 else 
    (CASE WHEN 
     count(distinct placement.id) = 0 THEN (count(distinct(case when 
     placement.commitmentstatus='fullfilled' then placement.id else 0 
    end)) / 1) 
    else (cast(count(distinct(case when 
    placement.commitmentstatus='fullfilled' then placement.id else 0 
    end)) as decimal(10,2)) / cast(count(distinct placement.id) as 
    decimal(10,2)))
    end) end as "COMMITMENT TO FULFILLMENT %"

from lpmysqldb.users u
left join lpmysqldb.teams t on t.team_id=u.team_id
left join lpmysqldb.organizations o on o.id=t.org_id
left join (select * from lpmysqldb.activity where 
  org_id='555b918ae4b07b6ac5050852' and completed_at>='2018-05-01' and 
  completed_at<='2018-06-01' and tag='visit' and accountname is not 
  null and (status='active' or status='true' or status='1')) as 
  activity on activity.user_id=u.id
left join (select * from lpmysqldb.placements where 
  orgid='555b918ae4b07b6ac5050852' and placementdate>='2018-05-01' and 
  placementdate<='2018-06-01' and (status IN ('1','active','true') or 
  status is null)) as placement on placement.userid=u.id

where u.org_id='555b918ae4b07b6ac5050852' 
  and (u.status='active' or u.status='true' or u.status='1')
  and istestuser!='1'
group by u.org_id, t.name, u.id, u.name, o.mode
order by count(distinct activity.id) desc

enter image description here

谢谢您的帮助!

我在下面进行了编辑,将两个底部连接从子查询上的连接更改为直接在表上的连接。仍然产生相同的结果。

enter image description here

1 个答案:

答案 0 :(得分:0)

这是对您的查询的稍作重组的查询。可能会简化,因为最后两个子查询都已针对您各自的计数和计数差异进行了预先汇总,因此您可以直接使用这些列名,而不必显示整个查询中嵌入的所有count(distance)。

我还尝试通过将给定计数乘以1.00来强制采用基于十进制的精度来简化除法。

select 
      t.name as team, 
      u.name as "REP NAME", 
      Activity.DistIdCnt as "TOTAL VISITS", 
      Activity.UniqAccountCnt as "UNIQUE VISITS",
      Placement.DistIdCnt as "COMMITMENTS ADDED",

      Placement.DistIdCnt / 
         CASE WHEN Activity.UniqAccountCnt = 0 
           THEN 1.00 
           ELSE Activity.UniqAccountCnt * 1.00 
           end as "UNIQUE VISIT TO COMMITMENT %",

      case when o.mode = 'basic' 
           then Placement.DistIdCnt
           else Placement.DistFulfillCnt
           end as "COMMITMENTS FULFILLED",

      case when o.mode = 'basic' 
           then 1 
           else ( Placement.DistFulfillCnt /
                     CASE when Placement.DistIdCnt = 0
                          then 1.00
                          ELSE Placement.DistIdCnt * 1.00
                          END TRANSACTION )
           END as "COMMITMENT TO FULFILLMENT %"
   from 
      lpmysqldb.users u
         left join lpmysqldb.teams t 
            on u.team_id = t.team_id
            left join lpmysqldb.organizations o 
               on t.org_id = o.id
        left join
        ( select
                user_id,
                count(*) as AllRecs,
                count( distinct id ) DistIdCnt,
                count( distinct account_id) as UniqAccountCnt
             from
                lpmysqldb.activity
             where 
                    org_id = '555b918ae4b07b6ac5050852' 
                and completed_at>='2018-05-01' 
                and completed_at<='2018-06-01' 
                and tag='visit' 
                and accountname is not null 
                and status IN ( '1', 'active', 'true') 
             group by
                user_id ) activity 
            on u.id = activity.user_id

         left join 
         ( select
                 userid,
                 count(*) AllRecs,
                 count(distinct id) as DistIdCnt,
                 count(distinct( case when commitmentstatus = 'fullfilled' 
                                      then id 
                                      else 0 end )) DistFulfillCnt
              from 
                 lpmysqldb.placements
              where 
                     orgid = '555b918ae4b07b6ac5050852'
                 and placementdate >= '2018-05-01' 
                 and placementdate <= '2018-06-01' 
                 and ( status is null OR status IN ('1','active','true')
              group by
                 userid ) as placement 
            on u.id = placement.userid
   where 
         u.org_id = '555b918ae4b07b6ac5050852' 
     and u.status IN ( 'active', 'true', '1')
     and istestuser != '1'
   group by 
      u.org_id, 
      t.name, 
      u.id, 
      u.name, 
      o.mode
   order by 
      activity.DistIdCnt desc

最后,您的内部查询正在查询所有用户。如果您有大量不活跃的用户,则可以通过在其中添加那些联接/条件来从每个内部查询中排除这些用户,例如...

( ...
              from 
                 lpmysqldb.placements 
                    JOIN lpmysqldb.users u2
                       on placements.userid = u2.id
                      and u2.status IN ( 'active', 'true', '1')
                      and u2.istestuser != '1'
              where … ) as placement