9

我在 SQL Server 2008 R2 中有一个客户和订单表。两者都有关于客户 ID 的索引(称为id)。我需要返回客户表中所有客户的详细信息以及订单表中的信息,例如第一个订单的详细信息。

我目前在订单表的子查询中加入了我的客户表,子查询返回我需要的有关订单的信息。例如:

SELECT c.id
        ,c.country      
        ,First_orders.product
        ,First_orders.order_id
FROM customers c

LEFT JOIN   SELECT( id, 
                    product 
            FROM (SELECT    id
                            ,product
                            ,order_id
                            ,ROW_NUMBER() OVER (PARTITION BY id ORDER BY Order_Date asc) as order_No 
                        FROM orders) orders
            WHERE Order_no = 1) First_Orders
ON c.id = First_orders.id

我对 SQL 很陌生,想了解我是否有效地执行此操作。我最终在一个选择查询中将很多这样的子查询加入到客户表中,并且可能需要数十分钟才能运行。

那么我这样做是有效的还是可以改进的?例如,我不确定我在订单表中的 id 索引是否有任何用途,也许我可以通过首先创建子查询中内容的临时表并在 id 上创建唯一索引来加快查询速度临时表,所以 SQL Server 知道id现在是一个唯一列,然后将我的客户表加入这个临时表?我通常在客户和订单表中有一两百万行。

提前谢谢了!

4

3 回答 3

7

您可以删除其中一个子查询以提高效率:

SELECT c.id
        ,c.country      
        ,First_orders.product
        ,First_orders.order_id
FROM customers c
   LEFT JOIN  (SELECT id
                    ,product
                    ,order_id
                    ,ROW_NUMBER() OVER (PARTITION BY id ORDER BY Order_Date asc) as order_No 
               FROM orders) First_Orders
     ON c.id = First_orders.id AND First_Orders.order_No = 1

在您的上述查询中,您需要小心放置括号的位置,因为我认为它不会起作用。此外,您在结果中返回产品,但不包括在嵌套子查询中。

于 2013-02-26T17:17:05.173 回答
2

对于刚刚学习 SQL 的人来说,您的查询看起来相当不错。

客户索引可能会也可能不会用于查询——您需要查看执行计划。索引 onorders(id, order_date)可以非常有效地用于该row_number功能。

一条评论是关于字段的命名。该字段orders.id不应是客户 ID。这应该类似于“orders.Customer_Id”。保持跨表的命名系统一致将在未来帮助您。

于 2013-02-26T17:16:25.983 回答
1

试试这个...它很容易理解

;WITH cte
AS (
    SELECT id
        ,product
        ,order_id
        ,ROW_NUMBER() OVER (
            PARTITION BY id ORDER BY Order_Date ASC
            ) AS order_No
    FROM orders
    )
SELECT c.id
    ,c.country
    ,c1.Product
    ,c1.order_id
FROM customers c
INNER JOIN cte c1 ON c.id = c1.id
WHERE c1.order_No = 1
于 2018-01-18T13:33:06.173 回答