将连接与聚合一起使用,并在不存在聚合时检索行

时间:2013-03-25 08:09:11

标签: mysql join aggregate-functions

我的MySQL表上的以下查询返回purchaseorder表中的行,这些行在deliveryorder表中具有相应的条目。如何构造此查询以便即使在deliveryorder表中不存在相应的行,我也会从purchaseorder表中获取行?如果用户想要查看sql表CREATE语句,我可以发布这些语句,但我现在不发帖,因为它确实使问题太大了。

SELECT
    `purchaseorder`.`id` AS `po_id`,
    `purchaseorder`.`order_quantity` AS `po_order_quantity`,
    `purchaseorder`.`applicable_approved_unit_rate` AS `po_unit_rate`,
    `purchaseorder`.`applicable_sales_tax_rate` AS `po_tax_rate`,
    `purchaseorder`.`order_date` AS `po_order_date`,
    `purchaseorder`.`remarks` AS `po_remarks`,
    `purchaseorder`.`is_open` AS `po_is_open`,
    `purchaseorder`.`is_active` AS `po_is_active`,
    `purchaseorder`.`approved_rate_id` AS `po_app_rate_id`,

    `supplier`.`name` AS `sup_name`,

    SUM(`deliveryorder`.`quantity`) AS `total_ordered`

    FROM `purchaseorder`
    LEFT JOIN `deliveryorder` ON (`deliveryorder`.`purchase_order_id` = `purchaseorder`.`id`)
    INNER JOIN `approvedrate` ON (`purchaseorder`.`approved_rate_id` = `approvedrate`.`id`)
    INNER JOIN `supplier` ON (`approvedrate`.`supplier_id` = `supplier`.`id`)

    WHERE (
            `purchaseorder`.`is_active` = 1
            AND `purchaseorder`.`is_open` = 1
            AND `deliveryorder`.`is_active` = 1
            AND `approvedrate`.`material_id` = 2
           )
    HAVING `purchaseorder`.`order_quantity` >= `total_ordered` + 1

2 个答案:

答案 0 :(得分:0)

你有一个聚合函数但没有GROUP BY子句,这很奇怪,但无论如何 - 这样的事情?哎呀 - 编辑......

SELECT po.id po_id
     , po.order_quantity po_order_quantity
     , po.applicable_approved_unit_rate po_unit_rate
     , po.applicable_sales_tax_rate po_tax_rate
     , po.order_date po_order_date
     , po.remarks po_remarks
     , po.is_open po_is_open
     , po.is_active po_is_active
     , po.approved_rate_id po_app_rate_id
     , s.name sup_name
     , SUM(do.quantity) total_ordered
  FROM purchaseorder po
  LEFT 
  JOIN deliveryorder do 
    ON do.purchase_order_id = po.  
   AND do.is_active = 1
  LEFT
  JOIN approvedrate ar
    ON ar.id = po.approved_rate_id 
   AND ar.material_id = 2
  LEFT
  JOIN supplier s
    ON s.id = ar.supplier_id 
 WHERE po.is_active = 1
   AND po.is_open = 1
HAVING po.order_quantity >= total_ordered + 1

答案 1 :(得分:0)

我无法弄清楚如何在一个查询中获得所需的结果,但最终使用以下两个查询来满足我的要求: -

第一次查询

        SELECT
            pot.`id` AS `po_id`,
            pot.`order_quantity` AS `po_order_quantity`,
            pot.`applicable_approved_unit_rate` AS `po_unit_rate`,
            pot.`applicable_sales_tax_rate` AS `po_tax_rate`,
            pot.`is_open` AS `po_is_open`,
            pot.`is_active` AS `po_is_active`,

            st.`id` AS `sup_id`,
            st.`name` AS `sup_name`,

            SUM(dot.`quantity`) AS `total_ordered`
        FROM `purchaseorder` pot
            INNER JOIN `deliveryorder` dot ON (dot.`purchase_order_id` = pot.`id`)
            INNER JOIN `approvedrate` art ON (pot.`approved_rate_id` = art.`id`)
            INNER JOIN `supplier` st ON (art.`supplier_id` = st.`id`)
        WHERE (
                pot.`is_active` = 1
                AND pot.`is_open` = 1
                AND art.`material_id` = @materialid
                AND art.`in_effect` = 1
                AND art.`is_active` = 1
                AND dot.`is_active` = 1
                AND st.`is_active` = 1
               )
        HAVING pot.`order_quantity` >= `total_ordered` + @materialquantity

第二次查询

        SELECT
            pot.`id` AS `po_id`,
            pot.`order_quantity` AS `po_order_quantity`,
            pot.`applicable_approved_unit_rate` AS `po_unit_rate`,
            pot.`applicable_sales_tax_rate` AS `po_tax_rate`,
            pot.`is_open` AS `po_is_open`,
            pot.`is_active` AS `po_is_active`,

            st.`id` AS `sup_id`,
            st.`name` AS `sup_name`,

            0 AS `total_ordered`
        FROM `purchaseorder` pot
            INNER JOIN `approvedrate` art ON (pot.`approved_rate_id` = art.`id`)
            INNER JOIN `supplier` st ON (art.`supplier_id` = st.`id`)
        WHERE (
                pot.`is_active` = 1
                AND pot.`is_open` = 1
                AND art.`material_id` = @materialid
                AND art.`in_effect` = 1
                AND art.`is_active` = 1
                AND st.`is_active` = 1
                AND pot.`order_quantity` >= @materialquantity
                AND pot.`id` NOT IN
                    (
                      SELECT dot.`purchase_order_id`
                      FROM `deliveryorder` dot
                      WHERE dot.is_active = 1
                    )
               )