COUNTing 函数输出 returns 与 运行 不同的值,直接稍微修改 SQL
COUNTing function output returns a different value than running slightly modified SQL directly
编辑:数据库是 SQL Server 2008R2
所以,基本上我遇到了行数 != 计数 (*) 的问题。 运行 此代码:
SELECT *
FROM Fn_ForecastReport_TEST(NULL)
Returns 519行输出。但是,运行宁此代码:
SELECT COUNT(*)
FROM Fn_ForecastReport_TEST(NULL)
Returns 502. 这非常令人费解,所以我把函数的代码:
CREATE FUNCTION dbo.Fn_ForecastReport_TEST(@Date DATE = NULL)
RETURNS @ForecastReportTable TABLE
(
InvcLine_SM NVARCHAR(155),
Cust_No NVARCHAR(15),
Cust_Name NVARCHAR(100),
Genus NVARCHAR(20),
S_City NVARCHAR(100),
[Default Whs] NVARCHAR(30),
U_CustChannel NVARCHAR(20),
[ActualKits-2] NUMERIC(38, 17),
[ActualKits-1] NUMERIC(38, 17),
ActualKits NUMERIC(38, 17),
[FC Kits] INT,
[FC Kits+1] INT,
[FC Kits+2] INT,
[FC Sales] NUMERIC(38, 6),
[FC Sales+1] NUMERIC(38, 6),
[FC Sales+2] NUMERIC(38, 6),
[Avg Price] NUMERIC(38, 6)
)
AS
BEGIN
-- If we provide NULL to the function, it will use today's date
SET @Date = ISNULL(@Date, GETDATE());
DECLARE
@TPlus2EndDate AS DATE,
@TPlus2StartDate AS DATE,
@TPlus1EndDate AS DATE,
@TPlus1StartDate AS DATE,
@TMinus0EndDate AS DATE,
@TMinus0StartDate AS DATE,
@TMinus1EndDate AS DATE,
@TMinus1StartDate AS DATE,
@TMinus2EndDate AS DATE,
@TMinus2StartDate AS DATE;
-- We use T Minus 0 as the basis for all other calculations so it needs to be calculated first
SELECT @TMinus0StartDate = FirstDay,
@TMinus0EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(@Date);
SELECT @TPlus2StartDate = FirstDay,
@TPlus2EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(DATEADD(MONTH, 2, @TMinus0StartDate));
SELECT @TPlus1StartDate = FirstDay,
@TPlus1EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(DATEADD(MONTH, 1, @TMinus0StartDate));
SELECT @TMinus1StartDate = FirstDay,
@TMinus1EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(DATEADD(MONTH, -1, @TMinus0StartDate));
SELECT @TMinus2StartDate = FirstDay,
@TMinus2EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(DATEADD(MONTH, -2, @TMinus0StartDate));
WITH Genera_CTE (Genus) AS (
SELECT '106' UNION ALL
SELECT '108' UNION ALL
SELECT '108YM' UNION ALL
SELECT '112' UNION ALL
SELECT '112XC' UNION ALL
SELECT '118'
),
SpecificPricesForEachBusinessPartnerByProductGenus AS (
SELECT T0.cardcode,
T1.U_Genus_Code,
sum((T1.U_NETcontents + 551) * T0.price) [Price]
FROM OSPP T0
INNER JOIN OITM T1 ON T0.ItemCode = T1.ItemCode
WHERE T0.price < 100
AND T1.U_Netcontents BETWEEN 450 AND 552
AND T1.Onhand <> 0
AND T1.U_Genus_Code IN (SELECT Genus FROM Genera_CTE)
GROUP BY T0.cardcode,
T1.U_Genus_Code
),
BestWarehouseForBusinessPartnerCity_CTE AS (
SELECT CardCode,
City,
U_Default_Whs
FROM CRD1
WHERE AdresType = 'S'
AND U_Default_Whs IS NOT NULL
GROUP BY CardCode,
City,
U_Default_Whs
),
MostRecentForecastsForPeriod AS (
SELECT *
FROM Fn_GetMostRecentForecastsForMonthOfDate(@Date)
UNION ALL
SELECT *
FROM Fn_GetMostRecentForecastsForMonthOfDate(@TPlus1StartDate)
UNION ALL
SELECT *
FROM Fn_GetMostRecentForecastsForMonthOfDate(@TPlus2StartDate)
),
ForecastsReformattedForReport AS (
SELECT T1.SlpName,
T0.U_Cust_ID,
T2.CardName,
T0.U_Genus,
T0.U_Ship_City,
'' [Dfl Whs],
T2.U_CustChannel [U_CustChannel],
0 [ActualKits],
0 [ActualKits-1],
0 [ActualKits-2],
SUM(CASE
WHEN T0.U_FC_Month BETWEEN @TMinus0StartDate AND @TMinus0EndDate
THEN T0.U_Sets
ELSE 0
END) [Forecast],
SUM(CASE
WHEN T0.U_FC_Month BETWEEN @TPlus1StartDate AND @TPlus1EndDate
THEN T0.U_Sets
ELSE 0
END) [Forecast+1],
SUM(CASE
WHEN T0.U_FC_Month BETWEEN @TPlus2StartDate AND @TPlus2EndDate
THEN T0.U_Sets
ELSE 0
END) [Forecast+2]
FROM MostRecentForecastsForPeriod T0
INNER JOIN OSLP T1 ON T0.U_SLP_ID = T1.SlpCode
INNER JOIN OCRD T2 ON T0.U_Cust_ID = T2.CardCode
WHERE T0.U_FC_Month BETWEEN @TMinus0StartDate AND @TPlus2EndDate
GROUP BY T1.SlpName,
T0.U_Cust_ID,
T2.CardName,
T0.U_Genus,
T0.U_Ship_City,
T2.U_CustChannel
),
ActualTransactionsFormattedForReport AS (
SELECT InvcLine_SM,
Cust_No,
Cust_Name,
Genus,
S_City,
CASE
WHEN U_Default_Whs IS NULL THEN 'Mesa Dflt'
ELSE U_Default_Whs
END AS 'Default Whs',
U_CustChannel,
SUM(CASE
WHEN Trx_Date BETWEEN @TMinus0StartDate AND @TMinus0EndDate
THEN Kits
ELSE 0
END) [ActualKits],
SUM(CASE
WHEN Trx_Date BETWEEN @TMinus1StartDate AND @TMinus1EndDate
THEN Kits
ELSE 0
END) [ActualKits-1],
SUM(CASE
WHEN Trx_Date BETWEEN @TMinus2StartDate AND @TMinus2EndDate
THEN Kits
ELSE 0
END) [ActualKits-2],
0 [Forecast],
0 [Forecast+1],
0 [Forecast+2]
FROM SWD_SALES_TRX
WHERE Genus IN (SELECT Genus FROM Genera_CTE)
AND Trx_Date BETWEEN @TMinus2StartDate AND @TMinus0EndDate
GROUP BY InvcLine_SM,
cust_no,
Cust_name,
U_Default_Whs,
Genus,
S_City,
U_CustChannel
),
TransactionsAndForecastsMergedForReport AS (
SELECT *
FROM ActualTransactionsFormattedForReport
UNION ALL
SELECT *
FROM ForecastsReformattedForReport
)
INSERT @ForecastReportTable
SELECT
T0.InvcLine_SM,
T0.Cust_No,
T0.Cust_Name,
T0.Genus,
T0.S_City,
CASE
WHEN MAX(A0.U_Default_Whs) = ''
OR MAX(A0.U_Default_Whs) IS NULL
THEN 'Mesa Dflt'
ELSE MAX(A0.U_Default_Whs)
END [Default Whs],
T0.U_CustChannel,
SUM(T0.[ActualKits-2]) [ActualKits-2],
SUM(T0.[ActualKits-1]) [ActualKits-1],
SUM(T0.[ActualKits]) [ActualKits],
Sum(T0.[Forecast]) [FC Kits],
Sum(T0.[Forecast+1]) [FC Kits+1],
Sum(T0.[Forecast+2]) [FC Kits+2],
SUM(T0.[Forecast]) * ISNULL(T5.Price, t6.total) [FC Sales],
SUM(T0.[Forecast+1]) * ISNULL(T5.Price, t6.total) [FC Sales+1],
SUM(T0.[Forecast+2]) * ISNULL(T5.Price, t6.total) [FC Sales+2],
MAX(T6.Total) [Avg Price]
FROM TransactionsAndForecastsMergedForReport T0
LEFT JOIN SpecificPricesForEachBusinessPartnerByProductGenus T5 ON T0.Cust_No = T5.CardCode
AND T5.U_Genus_Code = T0.Genus
LEFT JOIN (
SELECT T2.U_CustChannel,
T1.U_Genus_Code,
(CASE
WHEN SUM(CASE
WHEN T1.[InvntryUom] <> 'KIT'
THEN T0.Quantity
ELSE T0.Quantity * ISNULL(T1.U_NetContents, 1)
END) = 0
THEN 0
ELSE sum(T0.LineTotal) / SUM(CASE
WHEN T1.[InvntryUom] <> 'KIT'
THEN T0.Quantity
ELSE T0.Quantity * ISNULL(551 + T1.U_NetContents, 1)
END)
END) * (MAX(T1.U_NetContents) + 551) [Total]
FROM (
SELECT T0.CardCode,T1.LineTotal [LineTotal],T1.Quantity [Quantity],T1.ItemCode
FROM OINV T0
INNER JOIN INV1 T1 ON T0.DocEntry = T1.DocEntry
WHERE T0.Canceled = 'N'
AND (T1.LineTotal <> 0 OR TreeType = 'N')
UNION ALL
SELECT T0.CardCode,(T1.LineTotal) * -1 [LineTotal], -T1.Quantity [Quantity],T1.ItemCode
FROM ORIN T0
INNER JOIN RIN1 T1 ON T0.DocEntry = T1.DocEntry
WHERE T0.Canceled = 'N'
AND T0.docType <> 'S'
AND T0.CANCELED = 'N'
AND (T1.LineTotal <> 0 OR TreeType = 'N')
) T0
INNER JOIN OITM T1 ON T0.ItemCode = T1.ItemCode
INNER JOIN OCRD T2 ON T0.CardCode = T2.CardCode
WHERE T1.U_Netcontents BETWEEN 450 AND 552
AND t1.Onhand <> 0
AND (
T1.U_Genus_Code IN (SELECT Genus FROM Genera_CTE)
OR (T1.[InvntryUom] = 'KIT' AND T1.U_Genus_Code = '108')
)
GROUP BY T2.U_CustChannel,
T1.U_Genus_Code
) T6 ON T6.U_CustChannel = T0.U_CustChannel
AND T6.U_Genus_Code = T0.Genus
LEFT JOIN BestWarehouseForBusinessPartnerCity_CTE A0 ON A0.CardCode = T0.Cust_No
AND LOWER(A0.City) = LOWER(T0.S_City)
GROUP BY InvcLine_SM,
cust_no,
Cust_name,
genus,
S_City,
t5.price,
t6.total,
T0.U_CustChannel;
RETURN;
END
GO
我稍微修改了这个 SQL 以便它可以 运行 在函数之外,我注意到一些非常奇怪的东西,当我 运行 它时错误消失了函数外:
-- If we provide NULL to the function, it will use today's date
DECLARE @Date DATE = GETDATE();
DECLARE
@TPlus2EndDate AS DATE,
@TPlus2StartDate AS DATE,
@TPlus1EndDate AS DATE,
@TPlus1StartDate AS DATE,
@TMinus0EndDate AS DATE,
@TMinus0StartDate AS DATE,
@TMinus1EndDate AS DATE,
@TMinus1StartDate AS DATE,
@TMinus2EndDate AS DATE,
@TMinus2StartDate AS DATE;
-- We use T Minus 0 as the basis for all other calculations so it needs to be calculated first
SELECT @TMinus0StartDate = FirstDay,
@TMinus0EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(@Date);
SELECT @TPlus2StartDate = FirstDay,
@TPlus2EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(DATEADD(MONTH, 2, @TMinus0StartDate));
SELECT @TPlus1StartDate = FirstDay,
@TPlus1EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(DATEADD(MONTH, 1, @TMinus0StartDate));
SELECT @TMinus1StartDate = FirstDay,
@TMinus1EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(DATEADD(MONTH, -1, @TMinus0StartDate));
SELECT @TMinus2StartDate = FirstDay,
@TMinus2EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(DATEADD(MONTH, -2, @TMinus0StartDate));
WITH Genera_CTE (Genus) AS (
SELECT '106' UNION ALL
SELECT '108' UNION ALL
SELECT '108YM' UNION ALL
SELECT '112' UNION ALL
SELECT '112XC' UNION ALL
SELECT '118'
),
SpecificPricesForEachBusinessPartnerByProductGenus AS (
SELECT T0.cardcode,
T1.U_Genus_Code,
sum((T1.U_NETcontents + 551) * T0.price) [Price]
FROM OSPP T0
INNER JOIN OITM T1 ON T0.ItemCode = T1.ItemCode
WHERE T0.price < 100
AND T1.U_Netcontents BETWEEN 450 AND 552
AND T1.Onhand <> 0
AND T1.U_Genus_Code IN (SELECT Genus FROM Genera_CTE)
GROUP BY T0.cardcode,
T1.U_Genus_Code
),
BestWarehouseForBusinessPartnerCity_CTE AS (
SELECT CardCode,
City,
U_Default_Whs
FROM CRD1
WHERE AdresType = 'S'
AND U_Default_Whs IS NOT NULL
GROUP BY CardCode,
City,
U_Default_Whs
),
MostRecentForecastsForPeriod AS (
SELECT *
FROM Fn_GetMostRecentForecastsForMonthOfDate(@Date)
UNION ALL
SELECT *
FROM Fn_GetMostRecentForecastsForMonthOfDate(@TPlus1StartDate)
UNION ALL
SELECT *
FROM Fn_GetMostRecentForecastsForMonthOfDate(@TPlus2StartDate)
),
ForecastsReformattedForReport AS (
SELECT T1.SlpName,
T0.U_Cust_ID,
T2.CardName,
T0.U_Genus,
T0.U_Ship_City,
'' [Dfl Whs],
T2.U_CustChannel [U_CustChannel],
0 [ActualKits],
0 [ActualKits-1],
0 [ActualKits-2],
SUM(CASE
WHEN T0.U_FC_Month BETWEEN @TMinus0StartDate AND @TMinus0EndDate
THEN T0.U_Sets
ELSE 0
END) [Forecast],
SUM(CASE
WHEN T0.U_FC_Month BETWEEN @TPlus1StartDate AND @TPlus1EndDate
THEN T0.U_Sets
ELSE 0
END) [Forecast+1],
SUM(CASE
WHEN T0.U_FC_Month BETWEEN @TPlus2StartDate AND @TPlus2EndDate
THEN T0.U_Sets
ELSE 0
END) [Forecast+2]
FROM MostRecentForecastsForPeriod T0
INNER JOIN OSLP T1 ON T0.U_SLP_ID = T1.SlpCode
INNER JOIN OCRD T2 ON T0.U_Cust_ID = T2.CardCode
WHERE T0.U_FC_Month BETWEEN @TMinus0StartDate AND @TPlus2EndDate
GROUP BY T1.SlpName,
T0.U_Cust_ID,
T2.CardName,
T0.U_Genus,
T0.U_Ship_City,
T2.U_CustChannel
),
ActualTransactionsFormattedForReport AS (
SELECT InvcLine_SM,
Cust_No,
Cust_Name,
Genus,
S_City,
CASE
WHEN U_Default_Whs IS NULL THEN 'Mesa Dflt'
ELSE U_Default_Whs
END AS 'Default Whs',
U_CustChannel,
SUM(CASE
WHEN Trx_Date BETWEEN @TMinus0StartDate AND @TMinus0EndDate
THEN Kits
ELSE 0
END) [ActualKits],
SUM(CASE
WHEN Trx_Date BETWEEN @TMinus1StartDate AND @TMinus1EndDate
THEN Kits
ELSE 0
END) [ActualKits-1],
SUM(CASE
WHEN Trx_Date BETWEEN @TMinus2StartDate AND @TMinus2EndDate
THEN Kits
ELSE 0
END) [ActualKits-2],
0 [Forecast],
0 [Forecast+1],
0 [Forecast+2]
FROM SWD_SALES_TRX
WHERE Genus IN (SELECT Genus FROM Genera_CTE)
AND Trx_Date BETWEEN @TMinus2StartDate AND @TMinus0EndDate
GROUP BY InvcLine_SM,
cust_no,
Cust_name,
U_Default_Whs,
Genus,
S_City,
U_CustChannel
),
TransactionsAndForecastsMergedForReport AS (
SELECT *
FROM ActualTransactionsFormattedForReport
UNION ALL
SELECT *
FROM ForecastsReformattedForReport
),
FinalOutput AS (
SELECT T0.InvcLine_SM,
T0.Cust_No,
T0.Cust_Name,
T0.Genus,
T0.S_City,
CASE
WHEN MAX(A0.U_Default_Whs) = ''
OR MAX(A0.U_Default_Whs) IS NULL
THEN 'Mesa Dflt'
ELSE MAX(A0.U_Default_Whs)
END [Default Whs],
T0.U_CustChannel,
SUM(T0.[ActualKits-2]) [ActualKits-2],
SUM(T0.[ActualKits-1]) [ActualKits-1],
SUM(T0.[ActualKits]) [ActualKits],
Sum(T0.[Forecast]) [FC Kits],
Sum(T0.[Forecast+1]) [FC Kits+1],
Sum(T0.[Forecast+2]) [FC Kits+2],
SUM(T0.[Forecast]) * ISNULL(T5.Price, t6.total) [FC Sales],
SUM(T0.[Forecast+1]) * ISNULL(T5.Price, t6.total) [FC Sales+1],
SUM(T0.[Forecast+2]) * ISNULL(T5.Price, t6.total) [FC Sales+2],
MAX(T6.Total) [Avg Price]
FROM TransactionsAndForecastsMergedForReport T0
LEFT JOIN SpecificPricesForEachBusinessPartnerByProductGenus T5 ON T0.Cust_No = T5.CardCode
AND T5.U_Genus_Code = T0.Genus
LEFT JOIN (
SELECT T2.U_CustChannel,
T1.U_Genus_Code,
(CASE
WHEN SUM(CASE
WHEN T1.[InvntryUom] <> 'KIT'
THEN T0.Quantity
ELSE T0.Quantity * ISNULL(T1.U_NetContents, 1)
END) = 0
THEN 0
ELSE sum(T0.LineTotal) / SUM(CASE
WHEN T1.[InvntryUom] <> 'KIT'
THEN T0.Quantity
ELSE T0.Quantity * ISNULL(551 + T1.U_NetContents, 1)
END)
END) * (MAX(T1.U_NetContents) + 551) [Total]
FROM (
SELECT T0.CardCode,T1.LineTotal [LineTotal],T1.Quantity [Quantity],T1.ItemCode
FROM OINV T0
INNER JOIN INV1 T1 ON T0.DocEntry = T1.DocEntry
WHERE T0.Canceled = 'N'
AND (T1.LineTotal <> 0 OR TreeType = 'N')
UNION ALL
SELECT T0.CardCode,(T1.LineTotal) * -1 [LineTotal], -T1.Quantity [Quantity],T1.ItemCode
FROM ORIN T0
INNER JOIN RIN1 T1 ON T0.DocEntry = T1.DocEntry
WHERE T0.Canceled = 'N'
AND T0.docType <> 'S'
AND T0.CANCELED = 'N'
AND (T1.LineTotal <> 0 OR TreeType = 'N')
) T0
INNER JOIN OITM T1 ON T0.ItemCode = T1.ItemCode
INNER JOIN OCRD T2 ON T0.CardCode = T2.CardCode
WHERE T1.U_Netcontents BETWEEN 450 AND 552
AND t1.Onhand <> 0
AND (
T1.U_Genus_Code IN (SELECT Genus FROM Genera_CTE)
OR (T1.[InvntryUom] = 'KIT' AND T1.U_Genus_Code = '108')
)
GROUP BY T2.U_CustChannel,
T1.U_Genus_Code
) T6 ON T6.U_CustChannel = T0.U_CustChannel
AND T6.U_Genus_Code = T0.Genus
LEFT JOIN BestWarehouseForBusinessPartnerCity_CTE A0 ON A0.CardCode = T0.Cust_No
AND LOWER(A0.City) = LOWER(T0.S_City)
GROUP BY InvcLine_SM,
cust_no,
Cust_name,
genus,
S_City,
t5.price,
t6.total,
T0.U_CustChannel
)
SELECT count(*)
FROM FinalOutput;
基本上,唯一的变化是将@Date 切换为 DECLARE 而不是 SET,同时删除函数输出的 INSERT。是否有一些函数导致计数 return 与实际行数不同的值?
这是 JetBrain 的 DataGrip 产品的问题。当我 运行 在 SSMS 中使用相同的代码时,它返回了正确的 COUNT 数量。
SQL 服务器有时会提供不准确的信息 sys.sysindexes table/view...尝试更新您的索引和统计信息,看看它是否仍然不匹配。
如果它被清除,那么 count(*) 会从索引而不是表中提取行计数,而 select 返回实际行。
nvm,jetbrain 做到了!好厉害!!
编辑:数据库是 SQL Server 2008R2
所以,基本上我遇到了行数 != 计数 (*) 的问题。 运行 此代码:
SELECT *
FROM Fn_ForecastReport_TEST(NULL)
Returns 519行输出。但是,运行宁此代码:
SELECT COUNT(*)
FROM Fn_ForecastReport_TEST(NULL)
Returns 502. 这非常令人费解,所以我把函数的代码:
CREATE FUNCTION dbo.Fn_ForecastReport_TEST(@Date DATE = NULL)
RETURNS @ForecastReportTable TABLE
(
InvcLine_SM NVARCHAR(155),
Cust_No NVARCHAR(15),
Cust_Name NVARCHAR(100),
Genus NVARCHAR(20),
S_City NVARCHAR(100),
[Default Whs] NVARCHAR(30),
U_CustChannel NVARCHAR(20),
[ActualKits-2] NUMERIC(38, 17),
[ActualKits-1] NUMERIC(38, 17),
ActualKits NUMERIC(38, 17),
[FC Kits] INT,
[FC Kits+1] INT,
[FC Kits+2] INT,
[FC Sales] NUMERIC(38, 6),
[FC Sales+1] NUMERIC(38, 6),
[FC Sales+2] NUMERIC(38, 6),
[Avg Price] NUMERIC(38, 6)
)
AS
BEGIN
-- If we provide NULL to the function, it will use today's date
SET @Date = ISNULL(@Date, GETDATE());
DECLARE
@TPlus2EndDate AS DATE,
@TPlus2StartDate AS DATE,
@TPlus1EndDate AS DATE,
@TPlus1StartDate AS DATE,
@TMinus0EndDate AS DATE,
@TMinus0StartDate AS DATE,
@TMinus1EndDate AS DATE,
@TMinus1StartDate AS DATE,
@TMinus2EndDate AS DATE,
@TMinus2StartDate AS DATE;
-- We use T Minus 0 as the basis for all other calculations so it needs to be calculated first
SELECT @TMinus0StartDate = FirstDay,
@TMinus0EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(@Date);
SELECT @TPlus2StartDate = FirstDay,
@TPlus2EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(DATEADD(MONTH, 2, @TMinus0StartDate));
SELECT @TPlus1StartDate = FirstDay,
@TPlus1EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(DATEADD(MONTH, 1, @TMinus0StartDate));
SELECT @TMinus1StartDate = FirstDay,
@TMinus1EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(DATEADD(MONTH, -1, @TMinus0StartDate));
SELECT @TMinus2StartDate = FirstDay,
@TMinus2EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(DATEADD(MONTH, -2, @TMinus0StartDate));
WITH Genera_CTE (Genus) AS (
SELECT '106' UNION ALL
SELECT '108' UNION ALL
SELECT '108YM' UNION ALL
SELECT '112' UNION ALL
SELECT '112XC' UNION ALL
SELECT '118'
),
SpecificPricesForEachBusinessPartnerByProductGenus AS (
SELECT T0.cardcode,
T1.U_Genus_Code,
sum((T1.U_NETcontents + 551) * T0.price) [Price]
FROM OSPP T0
INNER JOIN OITM T1 ON T0.ItemCode = T1.ItemCode
WHERE T0.price < 100
AND T1.U_Netcontents BETWEEN 450 AND 552
AND T1.Onhand <> 0
AND T1.U_Genus_Code IN (SELECT Genus FROM Genera_CTE)
GROUP BY T0.cardcode,
T1.U_Genus_Code
),
BestWarehouseForBusinessPartnerCity_CTE AS (
SELECT CardCode,
City,
U_Default_Whs
FROM CRD1
WHERE AdresType = 'S'
AND U_Default_Whs IS NOT NULL
GROUP BY CardCode,
City,
U_Default_Whs
),
MostRecentForecastsForPeriod AS (
SELECT *
FROM Fn_GetMostRecentForecastsForMonthOfDate(@Date)
UNION ALL
SELECT *
FROM Fn_GetMostRecentForecastsForMonthOfDate(@TPlus1StartDate)
UNION ALL
SELECT *
FROM Fn_GetMostRecentForecastsForMonthOfDate(@TPlus2StartDate)
),
ForecastsReformattedForReport AS (
SELECT T1.SlpName,
T0.U_Cust_ID,
T2.CardName,
T0.U_Genus,
T0.U_Ship_City,
'' [Dfl Whs],
T2.U_CustChannel [U_CustChannel],
0 [ActualKits],
0 [ActualKits-1],
0 [ActualKits-2],
SUM(CASE
WHEN T0.U_FC_Month BETWEEN @TMinus0StartDate AND @TMinus0EndDate
THEN T0.U_Sets
ELSE 0
END) [Forecast],
SUM(CASE
WHEN T0.U_FC_Month BETWEEN @TPlus1StartDate AND @TPlus1EndDate
THEN T0.U_Sets
ELSE 0
END) [Forecast+1],
SUM(CASE
WHEN T0.U_FC_Month BETWEEN @TPlus2StartDate AND @TPlus2EndDate
THEN T0.U_Sets
ELSE 0
END) [Forecast+2]
FROM MostRecentForecastsForPeriod T0
INNER JOIN OSLP T1 ON T0.U_SLP_ID = T1.SlpCode
INNER JOIN OCRD T2 ON T0.U_Cust_ID = T2.CardCode
WHERE T0.U_FC_Month BETWEEN @TMinus0StartDate AND @TPlus2EndDate
GROUP BY T1.SlpName,
T0.U_Cust_ID,
T2.CardName,
T0.U_Genus,
T0.U_Ship_City,
T2.U_CustChannel
),
ActualTransactionsFormattedForReport AS (
SELECT InvcLine_SM,
Cust_No,
Cust_Name,
Genus,
S_City,
CASE
WHEN U_Default_Whs IS NULL THEN 'Mesa Dflt'
ELSE U_Default_Whs
END AS 'Default Whs',
U_CustChannel,
SUM(CASE
WHEN Trx_Date BETWEEN @TMinus0StartDate AND @TMinus0EndDate
THEN Kits
ELSE 0
END) [ActualKits],
SUM(CASE
WHEN Trx_Date BETWEEN @TMinus1StartDate AND @TMinus1EndDate
THEN Kits
ELSE 0
END) [ActualKits-1],
SUM(CASE
WHEN Trx_Date BETWEEN @TMinus2StartDate AND @TMinus2EndDate
THEN Kits
ELSE 0
END) [ActualKits-2],
0 [Forecast],
0 [Forecast+1],
0 [Forecast+2]
FROM SWD_SALES_TRX
WHERE Genus IN (SELECT Genus FROM Genera_CTE)
AND Trx_Date BETWEEN @TMinus2StartDate AND @TMinus0EndDate
GROUP BY InvcLine_SM,
cust_no,
Cust_name,
U_Default_Whs,
Genus,
S_City,
U_CustChannel
),
TransactionsAndForecastsMergedForReport AS (
SELECT *
FROM ActualTransactionsFormattedForReport
UNION ALL
SELECT *
FROM ForecastsReformattedForReport
)
INSERT @ForecastReportTable
SELECT
T0.InvcLine_SM,
T0.Cust_No,
T0.Cust_Name,
T0.Genus,
T0.S_City,
CASE
WHEN MAX(A0.U_Default_Whs) = ''
OR MAX(A0.U_Default_Whs) IS NULL
THEN 'Mesa Dflt'
ELSE MAX(A0.U_Default_Whs)
END [Default Whs],
T0.U_CustChannel,
SUM(T0.[ActualKits-2]) [ActualKits-2],
SUM(T0.[ActualKits-1]) [ActualKits-1],
SUM(T0.[ActualKits]) [ActualKits],
Sum(T0.[Forecast]) [FC Kits],
Sum(T0.[Forecast+1]) [FC Kits+1],
Sum(T0.[Forecast+2]) [FC Kits+2],
SUM(T0.[Forecast]) * ISNULL(T5.Price, t6.total) [FC Sales],
SUM(T0.[Forecast+1]) * ISNULL(T5.Price, t6.total) [FC Sales+1],
SUM(T0.[Forecast+2]) * ISNULL(T5.Price, t6.total) [FC Sales+2],
MAX(T6.Total) [Avg Price]
FROM TransactionsAndForecastsMergedForReport T0
LEFT JOIN SpecificPricesForEachBusinessPartnerByProductGenus T5 ON T0.Cust_No = T5.CardCode
AND T5.U_Genus_Code = T0.Genus
LEFT JOIN (
SELECT T2.U_CustChannel,
T1.U_Genus_Code,
(CASE
WHEN SUM(CASE
WHEN T1.[InvntryUom] <> 'KIT'
THEN T0.Quantity
ELSE T0.Quantity * ISNULL(T1.U_NetContents, 1)
END) = 0
THEN 0
ELSE sum(T0.LineTotal) / SUM(CASE
WHEN T1.[InvntryUom] <> 'KIT'
THEN T0.Quantity
ELSE T0.Quantity * ISNULL(551 + T1.U_NetContents, 1)
END)
END) * (MAX(T1.U_NetContents) + 551) [Total]
FROM (
SELECT T0.CardCode,T1.LineTotal [LineTotal],T1.Quantity [Quantity],T1.ItemCode
FROM OINV T0
INNER JOIN INV1 T1 ON T0.DocEntry = T1.DocEntry
WHERE T0.Canceled = 'N'
AND (T1.LineTotal <> 0 OR TreeType = 'N')
UNION ALL
SELECT T0.CardCode,(T1.LineTotal) * -1 [LineTotal], -T1.Quantity [Quantity],T1.ItemCode
FROM ORIN T0
INNER JOIN RIN1 T1 ON T0.DocEntry = T1.DocEntry
WHERE T0.Canceled = 'N'
AND T0.docType <> 'S'
AND T0.CANCELED = 'N'
AND (T1.LineTotal <> 0 OR TreeType = 'N')
) T0
INNER JOIN OITM T1 ON T0.ItemCode = T1.ItemCode
INNER JOIN OCRD T2 ON T0.CardCode = T2.CardCode
WHERE T1.U_Netcontents BETWEEN 450 AND 552
AND t1.Onhand <> 0
AND (
T1.U_Genus_Code IN (SELECT Genus FROM Genera_CTE)
OR (T1.[InvntryUom] = 'KIT' AND T1.U_Genus_Code = '108')
)
GROUP BY T2.U_CustChannel,
T1.U_Genus_Code
) T6 ON T6.U_CustChannel = T0.U_CustChannel
AND T6.U_Genus_Code = T0.Genus
LEFT JOIN BestWarehouseForBusinessPartnerCity_CTE A0 ON A0.CardCode = T0.Cust_No
AND LOWER(A0.City) = LOWER(T0.S_City)
GROUP BY InvcLine_SM,
cust_no,
Cust_name,
genus,
S_City,
t5.price,
t6.total,
T0.U_CustChannel;
RETURN;
END
GO
我稍微修改了这个 SQL 以便它可以 运行 在函数之外,我注意到一些非常奇怪的东西,当我 运行 它时错误消失了函数外:
-- If we provide NULL to the function, it will use today's date
DECLARE @Date DATE = GETDATE();
DECLARE
@TPlus2EndDate AS DATE,
@TPlus2StartDate AS DATE,
@TPlus1EndDate AS DATE,
@TPlus1StartDate AS DATE,
@TMinus0EndDate AS DATE,
@TMinus0StartDate AS DATE,
@TMinus1EndDate AS DATE,
@TMinus1StartDate AS DATE,
@TMinus2EndDate AS DATE,
@TMinus2StartDate AS DATE;
-- We use T Minus 0 as the basis for all other calculations so it needs to be calculated first
SELECT @TMinus0StartDate = FirstDay,
@TMinus0EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(@Date);
SELECT @TPlus2StartDate = FirstDay,
@TPlus2EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(DATEADD(MONTH, 2, @TMinus0StartDate));
SELECT @TPlus1StartDate = FirstDay,
@TPlus1EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(DATEADD(MONTH, 1, @TMinus0StartDate));
SELECT @TMinus1StartDate = FirstDay,
@TMinus1EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(DATEADD(MONTH, -1, @TMinus0StartDate));
SELECT @TMinus2StartDate = FirstDay,
@TMinus2EndDate = LastDay
FROM Fn_GetFirstAndLastDaysOfMonthFromDate(DATEADD(MONTH, -2, @TMinus0StartDate));
WITH Genera_CTE (Genus) AS (
SELECT '106' UNION ALL
SELECT '108' UNION ALL
SELECT '108YM' UNION ALL
SELECT '112' UNION ALL
SELECT '112XC' UNION ALL
SELECT '118'
),
SpecificPricesForEachBusinessPartnerByProductGenus AS (
SELECT T0.cardcode,
T1.U_Genus_Code,
sum((T1.U_NETcontents + 551) * T0.price) [Price]
FROM OSPP T0
INNER JOIN OITM T1 ON T0.ItemCode = T1.ItemCode
WHERE T0.price < 100
AND T1.U_Netcontents BETWEEN 450 AND 552
AND T1.Onhand <> 0
AND T1.U_Genus_Code IN (SELECT Genus FROM Genera_CTE)
GROUP BY T0.cardcode,
T1.U_Genus_Code
),
BestWarehouseForBusinessPartnerCity_CTE AS (
SELECT CardCode,
City,
U_Default_Whs
FROM CRD1
WHERE AdresType = 'S'
AND U_Default_Whs IS NOT NULL
GROUP BY CardCode,
City,
U_Default_Whs
),
MostRecentForecastsForPeriod AS (
SELECT *
FROM Fn_GetMostRecentForecastsForMonthOfDate(@Date)
UNION ALL
SELECT *
FROM Fn_GetMostRecentForecastsForMonthOfDate(@TPlus1StartDate)
UNION ALL
SELECT *
FROM Fn_GetMostRecentForecastsForMonthOfDate(@TPlus2StartDate)
),
ForecastsReformattedForReport AS (
SELECT T1.SlpName,
T0.U_Cust_ID,
T2.CardName,
T0.U_Genus,
T0.U_Ship_City,
'' [Dfl Whs],
T2.U_CustChannel [U_CustChannel],
0 [ActualKits],
0 [ActualKits-1],
0 [ActualKits-2],
SUM(CASE
WHEN T0.U_FC_Month BETWEEN @TMinus0StartDate AND @TMinus0EndDate
THEN T0.U_Sets
ELSE 0
END) [Forecast],
SUM(CASE
WHEN T0.U_FC_Month BETWEEN @TPlus1StartDate AND @TPlus1EndDate
THEN T0.U_Sets
ELSE 0
END) [Forecast+1],
SUM(CASE
WHEN T0.U_FC_Month BETWEEN @TPlus2StartDate AND @TPlus2EndDate
THEN T0.U_Sets
ELSE 0
END) [Forecast+2]
FROM MostRecentForecastsForPeriod T0
INNER JOIN OSLP T1 ON T0.U_SLP_ID = T1.SlpCode
INNER JOIN OCRD T2 ON T0.U_Cust_ID = T2.CardCode
WHERE T0.U_FC_Month BETWEEN @TMinus0StartDate AND @TPlus2EndDate
GROUP BY T1.SlpName,
T0.U_Cust_ID,
T2.CardName,
T0.U_Genus,
T0.U_Ship_City,
T2.U_CustChannel
),
ActualTransactionsFormattedForReport AS (
SELECT InvcLine_SM,
Cust_No,
Cust_Name,
Genus,
S_City,
CASE
WHEN U_Default_Whs IS NULL THEN 'Mesa Dflt'
ELSE U_Default_Whs
END AS 'Default Whs',
U_CustChannel,
SUM(CASE
WHEN Trx_Date BETWEEN @TMinus0StartDate AND @TMinus0EndDate
THEN Kits
ELSE 0
END) [ActualKits],
SUM(CASE
WHEN Trx_Date BETWEEN @TMinus1StartDate AND @TMinus1EndDate
THEN Kits
ELSE 0
END) [ActualKits-1],
SUM(CASE
WHEN Trx_Date BETWEEN @TMinus2StartDate AND @TMinus2EndDate
THEN Kits
ELSE 0
END) [ActualKits-2],
0 [Forecast],
0 [Forecast+1],
0 [Forecast+2]
FROM SWD_SALES_TRX
WHERE Genus IN (SELECT Genus FROM Genera_CTE)
AND Trx_Date BETWEEN @TMinus2StartDate AND @TMinus0EndDate
GROUP BY InvcLine_SM,
cust_no,
Cust_name,
U_Default_Whs,
Genus,
S_City,
U_CustChannel
),
TransactionsAndForecastsMergedForReport AS (
SELECT *
FROM ActualTransactionsFormattedForReport
UNION ALL
SELECT *
FROM ForecastsReformattedForReport
),
FinalOutput AS (
SELECT T0.InvcLine_SM,
T0.Cust_No,
T0.Cust_Name,
T0.Genus,
T0.S_City,
CASE
WHEN MAX(A0.U_Default_Whs) = ''
OR MAX(A0.U_Default_Whs) IS NULL
THEN 'Mesa Dflt'
ELSE MAX(A0.U_Default_Whs)
END [Default Whs],
T0.U_CustChannel,
SUM(T0.[ActualKits-2]) [ActualKits-2],
SUM(T0.[ActualKits-1]) [ActualKits-1],
SUM(T0.[ActualKits]) [ActualKits],
Sum(T0.[Forecast]) [FC Kits],
Sum(T0.[Forecast+1]) [FC Kits+1],
Sum(T0.[Forecast+2]) [FC Kits+2],
SUM(T0.[Forecast]) * ISNULL(T5.Price, t6.total) [FC Sales],
SUM(T0.[Forecast+1]) * ISNULL(T5.Price, t6.total) [FC Sales+1],
SUM(T0.[Forecast+2]) * ISNULL(T5.Price, t6.total) [FC Sales+2],
MAX(T6.Total) [Avg Price]
FROM TransactionsAndForecastsMergedForReport T0
LEFT JOIN SpecificPricesForEachBusinessPartnerByProductGenus T5 ON T0.Cust_No = T5.CardCode
AND T5.U_Genus_Code = T0.Genus
LEFT JOIN (
SELECT T2.U_CustChannel,
T1.U_Genus_Code,
(CASE
WHEN SUM(CASE
WHEN T1.[InvntryUom] <> 'KIT'
THEN T0.Quantity
ELSE T0.Quantity * ISNULL(T1.U_NetContents, 1)
END) = 0
THEN 0
ELSE sum(T0.LineTotal) / SUM(CASE
WHEN T1.[InvntryUom] <> 'KIT'
THEN T0.Quantity
ELSE T0.Quantity * ISNULL(551 + T1.U_NetContents, 1)
END)
END) * (MAX(T1.U_NetContents) + 551) [Total]
FROM (
SELECT T0.CardCode,T1.LineTotal [LineTotal],T1.Quantity [Quantity],T1.ItemCode
FROM OINV T0
INNER JOIN INV1 T1 ON T0.DocEntry = T1.DocEntry
WHERE T0.Canceled = 'N'
AND (T1.LineTotal <> 0 OR TreeType = 'N')
UNION ALL
SELECT T0.CardCode,(T1.LineTotal) * -1 [LineTotal], -T1.Quantity [Quantity],T1.ItemCode
FROM ORIN T0
INNER JOIN RIN1 T1 ON T0.DocEntry = T1.DocEntry
WHERE T0.Canceled = 'N'
AND T0.docType <> 'S'
AND T0.CANCELED = 'N'
AND (T1.LineTotal <> 0 OR TreeType = 'N')
) T0
INNER JOIN OITM T1 ON T0.ItemCode = T1.ItemCode
INNER JOIN OCRD T2 ON T0.CardCode = T2.CardCode
WHERE T1.U_Netcontents BETWEEN 450 AND 552
AND t1.Onhand <> 0
AND (
T1.U_Genus_Code IN (SELECT Genus FROM Genera_CTE)
OR (T1.[InvntryUom] = 'KIT' AND T1.U_Genus_Code = '108')
)
GROUP BY T2.U_CustChannel,
T1.U_Genus_Code
) T6 ON T6.U_CustChannel = T0.U_CustChannel
AND T6.U_Genus_Code = T0.Genus
LEFT JOIN BestWarehouseForBusinessPartnerCity_CTE A0 ON A0.CardCode = T0.Cust_No
AND LOWER(A0.City) = LOWER(T0.S_City)
GROUP BY InvcLine_SM,
cust_no,
Cust_name,
genus,
S_City,
t5.price,
t6.total,
T0.U_CustChannel
)
SELECT count(*)
FROM FinalOutput;
基本上,唯一的变化是将@Date 切换为 DECLARE 而不是 SET,同时删除函数输出的 INSERT。是否有一些函数导致计数 return 与实际行数不同的值?
这是 JetBrain 的 DataGrip 产品的问题。当我 运行 在 SSMS 中使用相同的代码时,它返回了正确的 COUNT 数量。
SQL 服务器有时会提供不准确的信息 sys.sysindexes table/view...尝试更新您的索引和统计信息,看看它是否仍然不匹配。
如果它被清除,那么 count(*) 会从索引而不是表中提取行计数,而 select 返回实际行。
nvm,jetbrain 做到了!好厉害!!