SQL-запрос для сравнения продаж товара по месяцам
-
08-06-2019 - |
Вопрос
У меня есть ежемесячное представление базы данных о состоянии, на основе которого мне нужно создать отчет.Данные в представлении выглядят примерно так:
Category | Revenue | Yearh | Month
Bikes 10 000 2008 1
Bikes 12 000 2008 2
Bikes 12 000 2008 3
Bikes 15 000 2008 1
Bikes 11 000 2007 2
Bikes 11 500 2007 3
Bikes 15 400 2007 4
...И так далее
В представлении есть категория продукта, доход, год и месяц.Я хочу создать отчет, сравнивающий 2007 и 2008 годы, показывающий 0 для месяцев без продаж.Таким образом, отчет должен выглядеть примерно так:
Category | Month | Rev. This Year | Rev. Last Year
Bikes 1 10 000 0
Bikes 2 12 000 11 000
Bikes 3 12 000 11 500
Bikes 4 0 15 400
Главное, на что следует обратить внимание, это то, что продажи за месяц 1 были только в 2008 году и, следовательно, равны 0 за 2007 год.Кроме того, у месяца 4 нет продаж только в 2008 году, следовательно, 0, в то время как у него есть продажи в 2007 году и он все еще отображается.
Кроме того, отчет фактически составлен за финансовый год, поэтому я хотел бы иметь пустые столбцы с 0 в обоих, если, скажем, в пятом месяце не было продаж ни за 2007, ни за 2008 год.
Запрос, который я получил, выглядит примерно так:
SELECT
SP1.Program,
SP1.Year,
SP1.Month,
SP1.TotalRevenue,
IsNull(SP2.TotalRevenue, 0) AS LastYearTotalRevenue
FROM PVMonthlyStatusReport AS SP1
LEFT OUTER JOIN PVMonthlyStatusReport AS SP2 ON
SP1.Program = SP2.Program AND
SP2.Year = SP1.Year - 1 AND
SP1.Month = SP2.Month
WHERE
SP1.Program = 'Bikes' AND
SP1.Category = @Category AND
(SP1.Year >= @FinancialYear AND SP1.Year <= @FinancialYear + 1) AND
((SP1.Year = @FinancialYear AND SP1.Month > 6) OR
(SP1.Year = @FinancialYear + 1 AND SP1.Month <= 6))
ORDER BY SP1.Year, SP1.Month
Проблема с этим запросом заключается в том, что он не вернет четвертую строку в моем примере данных выше, поскольку в 2008 году у нас не было никаких продаж, но на самом деле они были в 2007 году.
Вероятно, это распространенный запрос / проблема, но мой SQL устарел после столь долгой интерфейсной разработки.Мы очень ценим любую помощь!
Кстати, я использую SQL 2005 для этого запроса, поэтому, если появятся какие-либо полезные новые функции, которые могли бы мне помочь, дайте мне знать.
Решение
Оператор Case - мой лучший друг в sql.Вам также понадобится таблица времени для генерации вашего 0 оборотов за оба месяца.
Предположения основаны на наличии следующих таблиц:
продажи:Категория | Доход | Год | Месяц
и
тм:Год|Месяц (заполняется всеми датами, необходимыми для составления отчетности)
Пример 1 без пустых строк:
select
Category
,month
,SUM(CASE WHEN YEAR = 2008 THEN Revenue ELSE 0 END) this_year
,SUM(CASE WHEN YEAR = 2007 THEN Revenue ELSE 0 END) last_year
from
sales
where
year in (2008,2007)
group by
Category
,month
ВОЗВРАТ:
Category | Month | Rev. This Year | Rev. Last Year
Bikes 1 10 000 0
Bikes 2 12 000 11 000
Bikes 3 12 000 11 500
Bikes 4 0 15 400
Пример 2 с пустыми строками:Я собираюсь использовать вложенный запрос (но другие могут и не использовать) и верну пустую строку для каждого комбинированного продукта и месяца за год.
select
fill.Category
,fill.month
,SUM(CASE WHEN YEAR = 2008 THEN Revenue ELSE 0 END) this_year
,SUM(CASE WHEN YEAR = 2007 THEN Revenue ELSE 0 END) last_year
from
sales
Right join (select distinct --try out left, right and cross joins to test results.
product
,year
,month
from
sales --this ideally would be from a products table
cross join tm
where
year in (2008,2007)) fill
where
fill.year in (2008,2007)
group by
fill.Category
,fill.month
ВОЗВРАТ:
Category | Month | Rev. This Year | Rev. Last Year
Bikes 1 10 000 0
Bikes 2 12 000 11 000
Bikes 3 12 000 11 500
Bikes 4 0 15 400
Bikes 5 0 0
Bikes 6 0 0
Bikes 7 0 0
Bikes 8 0 0
Обратите внимание, что большинство инструментов отчетности будут выполнять эту функцию перекрестных таблиц или матриц, и теперь, когда я думаю об этом, SQL Server 2005 имеет синтаксис pivot, который также будет выполнять это.
Вот некоторые дополнительные ресурсы.КЕЙС http://www.4guysfromrolla.com/webtech/102704-1.shtml СВОДНЫЙ ФАЙЛ SQL SERVER 2005 http://msdn.microsoft.com/en-us/library/ms177410.aspx
Другие советы
@Christian - редактор markdown - ТЬФУ;особенно, когда предварительный просмотр и окончательная версия вашего поста расходятся...@Christian - полное внешнее объединение - полное внешнее объединение отменяется тем фактом, что в предложении WHERE есть ссылки на SP1, а предложение WHERE применяется после ОБЪЕДИНЕНИЯ.Чтобы выполнить полное внешнее объединение с фильтрацией в одной из таблиц, вам нужно поместить ваше предложение WHERE в подзапрос, чтобы произошла фильтрация до того, как объедините, или попытайтесь встроить все ваши критерии WHERE в предложение JOIN ON, которое безумно уродливо.Ну, на самом деле нет красивого способа сделать это.
@Джонас:Учитывая это:
Кроме того, отчет фактически составлен за финансовый год - так что Мне бы хотелось иметь пустые столбцы с 0 в обоих, если, скажем, в пятом месяце не было продаж ни за 2007, ни за 2008 год.
и тот факт, что эта работа не может быть выполнена с помощью красивого запроса, я бы определенно попытался получить результаты, которые вы действительно хотите.Нет смысла создавать некрасивый запрос и даже не получать точных данных, которые вам действительно нужны.;)
Итак, я бы предложил сделать это в 5 шагов:
1.создайте временную таблицу в формате, которому вы хотите, чтобы ваши результаты соответствовали
2.заполните его двенадцатью строками, указав 1-12 в столбце месяц
3.обновите столбец "В этом году", используя вашу логику SP1
4.обновите столбец "Прошлый год", используя вашу логику SP2
5.выберите из временной таблицы
Конечно, я предполагаю, что я работаю, исходя из предположения, что вы можете создать хранимую процедуру для достижения этой цели.Технически вы могли бы запустить весь этот пакет встроенным, но такого рода уродство встречается очень редко.Если вы не можете создать SP, я предлагаю вам вернуться к полному внешнему соединению с помощью подзапроса, но это не даст вам строку, если в течение месяца не было продаж ни в том, ни в другом году.
Насчет уценки - Да, это расстраивает.Редактор сделал предварительный просмотр моей HTML-таблицы, но после публикации она исчезла - поэтому пришлось удалить все HTML-форматирование из публикации...
@kcrumley Я думаю, мы пришли к аналогичным выводам.Этот запрос легко становится по-настоящему уродливым.Я действительно решил это до прочтения вашего ответа, используя аналогичный (но все же другой подход).У меня есть доступ к созданию хранимых процедур и функций в базе данных отчетов.Я создал функцию с табличным значением, принимающую категорию продукта и финансовый год в качестве параметра.Исходя из этого, функция заполнит таблицу, содержащую 12 строк.Строки будут заполнены данными из представления, если таковые имеются в наличии, в противном случае строка будет иметь 0 значений.
Затем я объединяю две таблицы, возвращаемые функциями.Поскольку я знаю, что во всех таблицах будет двенадцать roves, выделить их намного проще, и я могу присоединиться по категории продукта и месяцу:
SELECT
SP1.Program,
SP1.Year,
SP1.Month,
SP1.TotalRevenue AS ThisYearRevenue,
SP2.TotalRevenue AS LastYearRevenue
FROM GetFinancialYear(@Category, 'First Look', 2008) AS SP1
RIGHT JOIN GetFinancialYear(@Category, 'First Look', 2007) AS SP2 ON
SP1.Program = SP2.Program AND
SP1.Month = SP2.Month
Я думаю, что ваш подход, вероятно, немного чище, поскольку функция GetFinancialYear довольно грязная!Но, по крайней мере, это работает - что пока меня радует ;)
Хитрость заключается в том, чтобы выполнить ПОЛНОЕ ОБЪЕДИНЕНИЕ с помощью ISNULL, чтобы получить объединенные столбцы из любой таблицы.Обычно я помещаю это в представление или производную таблицу, в противном случае вам также нужно использовать ISNULL в предложении WHERE .
SELECT
Program,
Month,
ThisYearTotalRevenue,
PriorYearTotalRevenue
FROM (
SELECT
ISNULL(ThisYear.Program, PriorYear.Program) as Program,
ISNULL(ThisYear.Month, PriorYear.Month),
ISNULL(ThisYear.TotalRevenue, 0) as ThisYearTotalRevenue,
ISNULL(PriorYear.TotalRevenue, 0) as PriorYearTotalRevenue
FROM (
SELECT Program, Month, SUM(TotalRevenue) as TotalRevenue
FROM PVMonthlyStatusReport
WHERE Year = @FinancialYear
GROUP BY Program, Month
) as ThisYear
FULL OUTER JOIN (
SELECT Program, Month, SUM(TotalRevenue) as TotalRevenue
FROM PVMonthlyStatusReport
WHERE Year = (@FinancialYear - 1)
GROUP BY Program, Month
) as PriorYear ON
ThisYear.Program = PriorYear.Program
AND ThisYear.Month = PriorYear.Month
) as Revenue
WHERE
Program = 'Bikes'
ORDER BY
Month
Это должно обеспечить вам ваши минимальные требования - строки с продажами либо в 2007, либо в 2008 году, либо и в том, и в другом.Чтобы получить строки без продаж ни за один год, вам просто нужно выполнить ВНУТРЕННЕЕ ОБЪЕДИНЕНИЕ с таблицей чисел 1-12 (вы делаете выпей что-нибудь из этого, не так ли?).
Я могу ошибаться, но разве вы не должны использовать полное внешнее соединение вместо просто левого соединения?Таким образом, вы будете получать "пустые" столбцы из обеих таблиц.
Используя pivot и Динамический Sql, мы можем достичь этого результата
SET NOCOUNT ON
IF OBJECT_ID('TEMPDB..#TEMP') IS NOT NULL
DROP TABLE #TEMP
;With cte(Category , Revenue , Yearh , [Month])
AS
(
SELECT 'Bikes', 10000, 2008,1 UNION ALL
SELECT 'Bikes', 12000, 2008,2 UNION ALL
SELECT 'Bikes', 12000, 2008,3 UNION ALL
SELECT 'Bikes', 15000, 2008,1 UNION ALL
SELECT 'Bikes', 11000, 2007,2 UNION ALL
SELECT 'Bikes', 11500, 2007,3 UNION ALL
SELECT 'Bikes', 15400, 2007,4
)
SELECT * INTO #Temp FROM cte
Declare @Column nvarchar(max),
@Column2 nvarchar(max),
@Sql nvarchar(max)
SELECT @Column=STUFF((SELECT DISTINCT ','+ 'ISNULL('+QUOTENAME(CAST(Yearh AS VArchar(10)))+','+'''0'''+')'+ 'AS '+ QUOTENAME(CAST(Yearh AS VArchar(10)))
FROM #Temp order by 1 desc FOR XML PATH ('')),1,1,'')
SELECT @Column2=STUFF((SELECT DISTINCT ','+ QUOTENAME(CAST(Yearh AS VArchar(10)))
FROM #Temp FOR XML PATH ('')),1,1,'')
SET @Sql= N'SELECT Category,[Month],'+ @Column +'FRom #Temp
PIVOT
(MIN(Revenue) FOR yearh IN ('+@Column2+')
) AS Pvt
'
EXEC(@Sql)
Print @Sql
Результат
Category Month 2008 2007
----------------------------------
Bikes 1 10000 0
Bikes 2 12000 11000
Bikes 3 12000 11500
Bikes 4 0 15400