3

我正在使用 Firebird 2.1,我需要一些帮助来优化这个查询:(也许通过用 JOINS 替换 IN-s 或其他东西来加快它,因为它非常慢)

SELECT ClientID, ClientType, ClientName 
FROM Clients 
WHERE 
    (
        AccessRights = 0 OR 
        OwnerUserID = :uid OR 
        (
            AccessRights = 2 AND 
            ClientID IN (SELECT ClientID 
                            FROM ClientRights 
                            WHERE UserID = :uid)
        )
    ) 
    AND ClientID IN (SELECT CC.ClientID 
                    FROM CaseClients CC 
                    WHERE CC.CaseID IN (SELECT DISTINCT CaseID 
                                        FROM TimeSheet 
                                        WHERE IsBilled = 0) 
                        AND CC.ClientToBill = 1 
                        AND (SELECT BillingType 
                                FROM Cases 
                                WHERE CaseID = CC.CaseID) = 2
    );

谢谢!

4

4 回答 4

1
SELECT ClientID, ClientType, ClientName FROM Clients 
WHERE 
(
    AccessRights = 0 OR 
    OwnerUserID = :uid OR 
    (
        AccessRights = 2 AND 
        EXISTS(SELECT * FROM ClientRights r WHERE r.UserID = :uid and r.ClientId=Clients.ClientID)
    )
) 
AND EXISTS(SELECT * 
                FROM CaseClients CC 
                WHERE 
                CC.ClientID=Clients.ClientID and
                   EXISTS(SELECT * FROM TimeSheet 
                          WHERE IsBilled = 0 and TimeSheet.CaseID=CC.CaseID) 
                    AND CC.ClientToBill = 1 
                    AND EXISTS(SELECT BillingType 
                            FROM Cases 
                            WHERE CaseID = CC.CaseID and BillingType=2)
);
于 2012-02-08T18:50:38.197 回答
0

这可能会加快速度,但确保索引正确也会有所帮助..

SELECT C.ClientID, C.ClientType, C.ClientName 
FROM Clients C
INNER JOIN CaseClients CC ON C.ClientID = CC.ClientID AND CC.ClientToBill = 1
INNER JOIN TimeSheet TS ON CC.CaseID = TS.CaseID AND TS.IsBilled = 0
INNER JOIN Cases CS ON CC.CaseID = CS.CaseId AND CS.BillingType = 2 
WHERE AccessRights = 0 
     OR OwnerUserID = :uid 
     OR 
        (
            AccessRights = 2 AND 
            ClientID IN (SELECT ClientID 
                            FROM ClientRights 
                            WHERE UserID = :uid)
        )
    ) 
于 2012-02-08T17:10:00.360 回答
0
SELECT DISTINCT
        c.ClientID
       ,c.ClientType
       ,c.ClientName
  FROM Clients c
 INNER JOIN CaseClients cc
    ON c.ClientID = cc.ClientID
 INNER JOIN TimeSheet ts
    ON cc.CaseID = ts.CaseID
 INNER JOIN Cases ca
    ON cc.CaseID = ca.CaseID
 WHERE ts.IsBilled = 0
   AND cc.ClientToBill = 1
   AND ca.BillingType = 2
   AND (c.AccessRights = 0 OR
        c.OwnerUserID = :uid OR
        c.AccessRights = 2 AND c.ClientID IN (SELECT cr.ClientID
                                                FROM ClientRights cr
                                               WHERE cr.UserID = :uid)
       )
于 2012-02-08T17:16:44.117 回答
0

在我看来,这会更整洁更快:

SELECT  ClientID, ClientType, ClientName 
FROM    Clients C, CaseClients CC, TimeSheet TS, Cases CA
WHERE   C.ClientID = CC.ClientID 
AND CC.CaseID = TS.CaseID 
AND CC.CaseID = CA.CaseID 
AND TS.IsBilled = 0
AND     CC.ClientToBill = 1 
AND     CA.BillingType = 2 
AND     (C.AccessRights = 0 OR C.OwnerUserID = :uid OR (C.AccessRights = 2 AND     C.ClientID IN 
(SELECT ClientID FROM ClientRights WHERE UserID = :uid))) 
于 2012-02-08T17:17:24.910 回答