Переписать MySQL запрос с подзапросами в объединения - PullRequest
1 голос
/ 25 октября 2010

Я написал довольно сложный запрос SQL, чтобы получить некоторую статистику о животных из базы данных выборки животных.Этот запрос включает в себя несколько подзапросов, и теперь я хотел бы посмотреть, можно ли каким-либо образом переписать этот запрос, чтобы использовать соединения вместо подзапросов.У меня есть смутная идея, что это может сократить время запроса.(сейчас около 23 секунд на Mac mini).

Вот запрос:

SELECT COUNT(DISTINCT a.AnimalID), TO_DAYS(a.VisitDate) AS day, 
   DATE_FORMAT(a.VisitDate, '%b %d %Y'), a.origin, 
   (
    SELECT COUNT(DISTINCT a.AnimalID)
           FROM samples AS a 
                JOIN
                custom_animals AS b 
                ON a.AnimalID = b.animal_id
                WHERE
                     b.organism = 2
                     AND
                         TO_DAYS(a.VisitDate) = day
    ) AS Goats, 
    (
     SELECT COUNT(DISTINCT a.AnimalID) 
            FROM samples AS a 
                 JOIN custom_animals AS b 
                 ON a.AnimalID = b.animal_id 
                 WHERE
                      b.organism = 2
                      AND 
                         b.sex = 'Female'
                      AND
                         TO_DAYS(a.VisitDate) = day
    ) AS GF,
    (
     SELECT COUNT(DISTINCT a.AnimalID) 
            FROM samples AS a 
                 JOIN custom_animals AS b 
                 ON a.AnimalID = b.animal_id 
                 WHERE 
                      b.organism = 3
                      AND
                         b.sex = 'Female'
                      AND
                         TO_DAYS(a.VisitDate) = day
    ) AS SF
    FROM
        samples AS a 
        JOIN custom_animals AS b 
        ON a.AnimalID = b.animal_id 
        WHERE
             project = 5
             AND
                AnimalID LIKE 'AVD%'
        GROUP BY
                TO_DAYS(a.VisitDate);

Благодаря ksogor мой запрос теперь намного быстрее;

SELECT  DATE_FORMAT(s.VisitDate, '%b %d %Y') AS date,
    s.origin,
    SUM(IF(project = 5 AND s.AnimalID LIKE 'AVD%', 1, 0)) AS sampled_animals, 
    SUM(IF(ca.organism = 2, 1, 0)) AS sampled_goats,
    SUM(IF(ca.organism = 2 AND ca.sex = 'Female', 1, 0)) AS female_goats,
    SUM(IF(ca.organism = 3 AND ca.sex = 'Female', 1, 0)) AS female_sheep
FROM samples s JOIN custom_animals ca ON s.AnimalID = ca.animal_id
GROUP BY date;

Мне по-прежнему нужно было бы сделать этот запрос отдельным s.AnimalID, хотя сейчас он считает образцы, взятые у этих животных, а не самих животных.У кого-нибудь есть идеи?


После дополнительной помощи от ksogor Теперь у меня отличный запрос:

SELECT  DATE_FORMAT(s.VisitDate, '%b %d %Y') AS date,
    s.origin,
    SUM(IF(project = 5 AND s.AnimalID LIKE 'AVD%', 1, 0)) AS sampled_animals, 
    SUM(IF(ca.organism = 2, 1, 0)) AS sampled_goats,
    SUM(IF(ca.organism = 2 AND ca.sex = 'Female', 1, 0)) AS female_goats,
    SUM(IF(ca.organism = 3 AND ca.sex = 'Female', 1, 0)) AS female_sheep
FROM (
    SELECT DISTINCT AnimalID AS AnimalID,
           VisitDate,
           origin,
           project
           FROM samples
) s 
JOIN custom_animals ca ON s.AnimalID = ca.animal_id
GROUP BY date;

1 Ответ

2 голосов
/ 25 октября 2010

Вы можете просто использовать операторы if или case, например:

SELECT  SUM(if(project = 5 AND AnimealID LIKE 'AVD%', 1, 0)) AS countbyproj, 
        TO_DAYS(s.VisitDate) AS day, 
        DATE_FORMAT(s.VisitDate, '%b %d %Y') AS date,
        s.origin,
        SUM(if(ca.organism = 2, 1, 0)) AS countGoats,
        SUM(if(ca.organism = 2 AND ca.sex = 'Female', 1, 0)) AS countGF,
        SUM(if(ca.organism = 3 AND ca.sex = 'Female', 1, 0)) AS countSF
FROM samples s JOIN custom_animals ca ON s.AnimalID = ca.animal_id
GROUP BY TO_DAYS(a.VisitDate);

Я не могу проверить запрос, я не знаю, какого результата вы ожидаете и какие таблицы /отношения, которые у вас есть, так что это только пример с идеей.

Если вам нужно рассчитывать unque AnimealID's для каждого дня:

SELECT SUM(byproj) AS countbyproj,
        day,
        date,
        origin,
        SUM(Goats) AS countGoats,
        SUM(GF) AS countGF,
        SUM(SF) AS countSF
FROM (
    SELECT  s.AnimealID,
            if(project = 5 AND AnimealID LIKE 'AVD%', 1, 0) AS byproj, 
            TO_DAYS(s.VisitDate) AS day, 
            DATE_FORMAT(s.VisitDate, '%b %d %Y') AS date,
            s.origin,
            if(ca.organism = 2, 1, 0)) AS Goats,
            if(ca.organism = 2 AND ca.sex = 'Female', 1, 0) AS GF,
            if(ca.organism = 3 AND ca.sex = 'Female', 1, 0) AS SF
    FROM samples s JOIN custom_animals ca ON s.AnimalID = ca.animal_id
    ) dataset
GROUP BY dataset.day, dataset.AnimealID;
...