简体   繁体   English

最大和几个联接的SQL

[英]sql with max and several joins

I am trying to get the last prices each customer paid for each product on mysql. 我试图获得每个客户在mysql上为每种产品支付的最新价格。 the following sql is not giving me the right data. 下面的SQL没有给我正确的数据。 the max(dateLasFullfillment) doesn't much the row value and its not even the max vsalue. max(dateLasFullfillment)并没有太大的行值,甚至没有最大vsalue。 its like the group by works before the max. 它就像最大之前的作品。

select 
    'item' AS type, soitem.productnum as 'SKU',
    (soitem.unitprice / right(uom.code, length(uom.code) - 2)) as unitPrice, 
    replace(customer.name, "#", "") AS priceList, 
    max(soitem.dateLastFulfillment) 
from 
    soitem
left join 
    so ON so.id = soitem.soid
left join
    customer on so.customerid = customer.id
left join 
    product on product.num = soitem.productnum
left join  
    uom on product.uomid = uom.id
where 
    soitem.dateLastFulfillment > now() - interval 6 month 
    and soitem.unitprice > 0 
    and so.statusid in (20, 25, 60)
group by 
    soitem.productnum, customer.name
order by 
    PriceList

Here are some Tables samples with expected results. 以下是一些具有预期结果的表样本。 the sql must start with select statement, no declare etc unless there is no other option. 除非没有其他选择,否则sql必须以select语句开头,没有声明等。

SO Table: SO表:

id  billToName  customerid  dateCompleted  dateCreated  dateIssued  num
1  Name1  1  6/27/18  6/23/18  6/23/18  ordernum1
2  Name1  1  7/15/18  7/10/18  7/10/18  ordernum2
3  Name1  1  7/29/18  7/20/18  7/20/18  ordernum3
4  Name2  2  6/31/2018  6/30/18  6/30/18  ordernum4
5  Name2  2  7/27/18  7/26/18  7/26/18  ordernum5
6  Name3  3  8/8/18  8/5/18  8/5/18  ordernum6
7  Name3  3  7/25/18  7/20/18  7/20/18  ordernum7

SOITEM table: SOITEM表:

id  soId  unitPrice  dateLastFulfillment  productId  productNum  statusId  uomId  qtyOrdered
1  1  10  6/27/18  1  SKU-1  50  11  3
2  1  20  6/27/18  2  SKU-2  50  12  5
3  1  30  6/27/18  3  SKU-3  50  13  6
4  2  11  7/15/18  1  SKU-1  50  11  11
5  2  21  7/15/18  2  SKU-2  50  12  44
6  2  31  7/15/18  3  SKU-3  50  13  5
7  3  12  7/29/18  1  SKU-1  50  11  5
8  3  22  7/29/18  2  SKU-2  50  12  6
9  4  23  6/31/2018  2  SKU-2  50  12  9
10  4  33  6/31/2018  3  SKU-3  50  13  12
11  5  24  7/27/18  2  SKU-2  50  12  14
12  5  34  7/27/18  3  SKU-3  50  13  35
13  6  25  8/8/18  2  SKU-2  50  12  22
14  6  35  8/8/18  3  SKU-3  50  13  55
15  7  26  7/25/18  2  SKU-2  50  12  22
16  7  36  7/25/18  3  SKU-3  50  13  11

PRODUCT table: 产品表:

num  uomid
SKU-1  11
SKU-2  12
SKU-3  13

UOM table: UOM表:

id  code
11  cs10
12  cs20
13  cs30

CUSTOMER table: 客户表:

ID  NAME
1  CUSTOMER1#
2  CUSTOMER2#
3  CUSTOMER3#

EXPECTED RESULTS: 预期成绩:

type  SKU  unitPrice  priceList  max(soitem.dateLastFulfillment)
item  SKU-1  1.2  customer1  7/29/18
item  SKU-2  1.1  customer1  7/29/18
item  SKU-3  1.03  customer1  7/15/18
item  SKU-2  1.2  customer2  7/27/18
item  SKU-3  1.13  customer2  7/27/18
item  SKU-2  1.25  customer3  8/8/18
item  SKU-3  1.17  customer3  8/8/18

Try this, just the first part of your code, and see what you get. 试试吧,只是代码的第一部分,然后看看您得到了什么。 Then paste in your other Joins etc. (removing first two old where items, and all of the group by) 然后粘贴您的其他Joins等。(删除前两个旧的where项目,以及所有分组依据)

-- compute this once, instead of for each row
Declare @now_minus_6mos as date = DATEADD(month, -6, GETDATE());
print @now_minus_6mos;

select 
    'item' AS type, soitem.productnum as 'SKU',
    (soitem.unitprice / 1 ) as unitPrice,   -- I do not have UOM, so simplify to be one
    -- I do not have Customer    replace(customer.name, "#", "") AS priceList, 
    (soitem.dateLastFulfillment) -- remove the max, since we are getting only the last one
from 
    (Select * From
        (Select
            productnum
            ,unitprice
            ,dateLastFulfillment
            ,Row_Number() Over(Partition By productnum  Order By dateLastFulfillment DESC) as dlfRow
         From soitem
         --move where filters here to reduce number of rows returned
         Where
             soitem.dateLastFulfillment > @now_minus_6mos 
         and soitem.unitprice > 0 
         ) aa
     Where dlfRow = 1
     ) soitem

You can do this with "ranking" function. 您可以使用“排名”功能进行此操作。 They doesn't exists in MySQL but you can imitate them. 它们在MySQL中不存在,但是您可以模仿它们。

See this post. 看到这篇文章。

Rank function in MySQL MySQL中的排名函数

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM