2017-01-22 71 views
0

對於家庭作業,我試圖在2012年1月使用AdventureWorks2014查找每位銷售人員的暢銷產品。使用SQL Server,我如何才能在AdventureWorks2014中找到每個銷售人員的暢銷產品?

這是我到目前爲止有:

SELECT  
    Person.Person.LastName, Person.Person.FirstName, 
    Person.Person.MiddleName, 
    Employee_1.JobTitle, 
    Sales.SalesPerson.SalesQuota, Sales.SalesOrderHeader.OrderDate, 
    Production.Product.Name, 
    SUM(distinct OrderQty) AS Expr2 
FROM 
    Sales.SalesOrderDetail 
INNER JOIN 
    Production.Product ON Sales.SalesOrderDetail.ProductID = Production.Product.ProductID 
INNER JOIN 
    Sales.SalesOrderHeader ON Sales.SalesOrderDetail.SalesOrderID = Sales.SalesOrderHeader.SalesOrderID 
          AND Sales.SalesOrderDetail.SalesOrderID = Sales.SalesOrderHeader.SalesOrderID 
          AND Sales.SalesOrderDetail.SalesOrderID = Sales.SalesOrderHeader.SalesOrderID 
INNER JOIN 
    Sales.SalesPerson ON Sales.SalesOrderHeader.SalesPersonID = Sales.SalesPerson.BusinessEntityID 
         AND Sales.SalesOrderHeader.SalesPersonID = Sales.SalesPerson.BusinessEntityID 
         AND Sales.SalesOrderHeader.SalesPersonID = Sales.SalesPerson.BusinessEntityID 
         AND Sales.SalesOrderHeader.SalesPersonID = Sales.SalesPerson.BusinessEntityID 
INNER JOIN 
    HumanResources.Employee AS Employee_1 
INNER JOIN 
    Person.Person ON Employee_1.BusinessEntityID = Person.Person.BusinessEntityID 
     ON Sales.SalesPerson.BusinessEntityID = Employee_1.BusinessEntityID 
     AND Sales.SalesPerson.BusinessEntityID = Employee_1.BusinessEntityID 
WHERE  
    (Sales.SalesOrderHeader.OrderDate BETWEEN '2012-01-01' AND '2012-01-31') 
GROUP BY 
    Person.Person.LastName, Person.Person.FirstName, Person.Person.MiddleName, 
    Employee_1.JobTitle, 
    Sales.SalesPerson.SalesQuota, Sales.SalesOrderHeader.OrderDate, 
    Production.Product.Name 
ORDER BY 
    Person.Person.LastName, Production.Product.Name 

我無法弄清楚如何添加orderqty所有爲每個單獨的產品。在前兩排結果中,我有同一人銷售的同一產品。我想將這些添加到一起,然後找到每個銷售人員擁有的前5名產品?

任何人都可以幫忙嗎?

+0

您需要使用RANK()和PARTITION BY。這裏有很多例子,比如www.mikesknowledgebase.com/pages/SQLServer/Rank.htm –

+0

@MikeGledhill [sql server'rank()'''row_number()'](http://stackoverflow.com/問題/ 7747327/sql-rank-versus-row-number) - 我想你的意思是'row_number()' – SqlZim

回答

0

我最初的目標是將聚合數據集減少到最少的所需數量表以執行聚合(SalesOrderHeader, SalesOrderDetail)並將其與附加信息(例如Person, Employee)結合到表中。

我包含在聚合子查詢的Product表,但它可以在聚合查詢後進行,並加入到產品編號代替(將其添加到group byselect後)。


有很多方法可以做到這一點,這裏有一些:

交叉apply版本:

select 
     p.LastName 
    , p.FirstName 
    , p.MiddleName 
    , e.JobTitle 
    , s.SalesQuota 
    , s.OrderDate 
    , s.ProductName 
    , s.TotalQty 
    from Person.Person as p on 
    inner join HumanResources.Employee as e on e.BusinessEntityID = p.BusinessEntityID 
    cross apply ( 
     select top 5 /* 5 rows */ 
     soh.SalesPersonID 
     , p.ProductName 
     , TotalQty = sum(OrderQty) 
     from Sales.SalesOrderHeader as soh 
     inner join Sales.SalesOrderDetail sod on soh.SalesOrderID = sod.SalesOrderID 
     inner join Production.Product as pr on sod.ProductID = pr.ProductID 
     where soh.OrderDate between '2012-01-01' and '2012-01-31' 
      and s.SalesPersonID = p.BusinessEntityID /* per person */ 
     group by soh.SalesPersonID, p.ProductName 
     order by sum(OrderQty) desc 
     /* ordered by sum(OrderQty) descending */ 
    ) s 

top with ties版本:

select top 5 with ties 
    p.LastName 
    , p.FirstName 
    , p.MiddleName 
    , e.JobTitle 
    , s.SalesQuota 
    , s.OrderDate 
    , s.ProductName 
    , s.TotalQty 
    from Person.Person as p on 
     inner join HumanResources.Employee as e on e.BusinessEntityID = p.BusinessEntityID 
     inner join ( 
     select 
      soh.SalesPersonID 
     , p.ProductName 
     , TotalQty = sum(OrderQty) 
     from Sales.SalesOrderHeader as soh 
      inner join Sales.SalesOrderDetail sod on soh.SalesOrderID = sod.SalesOrderID 
      inner join Production.Product as pr on sod.ProductID = pr.ProductID 
      where soh.OrderDate between '2012-01-01' and '2012-01-31' 
      group by soh.SalesPersonID, p.ProductName 
    ) on s.SalesPersonID = p.BusinessEntityID 
order by row_number() over (partition by p.BusinessEntityID order by s.TotalQty desc) 
/* returns all rows where row_number() over() evaluates to 1,2,3,4 ,or 5 */ 

common table expressionrow_number()版本:

with top5 as (
select 
     p.LastName 
    , p.FirstName 
    , p.MiddleName 
    , e.JobTitle 
    , s.SalesQuota 
    , s.OrderDate 
    , s.ProductName 
    , s.TotalQty 
    , rn=row_number() over (partition by p.BusinessEntityID order by s.TotalQty desc) 
    from Person.Person as p on 
    inner join HumanResources.Employee as e on e.BusinessEntityID = p.BusinessEntityID 
    inner join ( 
     select 
     soh.SalesPersonID 
     , p.ProductName 
     , TotalQty = sum(OrderQty) 
     from Sales.SalesOrderHeader as soh 
     inner join Sales.SalesOrderDetail sod on soh.SalesOrderID = sod.SalesOrderID 
     inner join Production.Product as pr on sod.ProductID = pr.ProductID 
     where soh.OrderDate between '2012-01-01' and '2012-01-31' 
     group by soh.SalesPersonID, p.ProductName 
    ) on s.SalesPersonID = p.BusinessEntityID 
) 

select 
     LastName 
    , FirstName 
    , MiddleName 
    , JobTitle 
    , SalesQuota 
    , OrderDate 
    , ProductName 
    , TotalQty 
    from top5 
    where rn < 6