У меня есть представление базы данных «Ежемесячный статус», на основе которого мне нужно построить отчет. Данные в представлении выглядят примерно так:
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 в обоих, если бы не было продаж, скажем, в 5-м месяце либо за 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 для этого запроса, поэтому, если есть какие-либо полезные новые функции, которые могут помочь мне, дайте мне знать.


Я могу ошибаться, но разве вы не должны использовать полное внешнее соединение вместо простого левого соединения? Таким образом вы получите «пустые» столбцы из обеих таблиц.
@Christian - редактор разметки - UGH; особенно когда предварительный просмотр и окончательная версия вашего поста расходятся ... @Christian - полное внешнее соединение - полное внешнее соединение отменяется тем фактом, что в предложении WHERE есть ссылки на SP1, а предложение WHERE применяется после JOIN. Чтобы выполнить полное внешнее соединение с фильтрацией в одной из таблиц, вам нужно поместить предложение WHERE в подзапрос, чтобы фильтрация происходила перед соединения, или попытаться построить все ваши критерии WHERE на предложении JOIN ON, которое является безумно уродливо. Что ж, на самом деле нет красивого способа сделать это.
@Jonas: Учитывая это:
Also, the report is actually for financial year - so I would love to have empty columns with 0 in both if there was no sales in say month 5 for either 2007 or 2008.
и тот факт, что эта работа не может быть выполнена с помощью красивого запроса, я определенно постараюсь получить те результаты, которые вам действительно нужны. Нет смысла делать уродливый запрос и даже не получать точные данные, которые вам действительно нужны. ;)
Итак, я предлагаю сделать это за 5 шагов:
1. создайте временную таблицу в формате, который вы хотите, чтобы ваши результаты соответствовали
2. Заполните его двенадцатью строками, от 1 до 12 в столбце месяца
.
3. обновите столбец «Этот год», используя логику SP1 logic
.
4. обновите столбец «Последний год», используя логику SP2
.
5. выберите из временной таблицы
Конечно, я полагаю, что исхожу из предположения, что для этого можно создать хранимую процедуру. Технически вы можете запустить весь этот пакет в режиме онлайн, но такое уродство встречается очень редко. Если вы не можете создать SP, я предлагаю вам вернуться к полному внешнему соединению с помощью подзапроса, но это не принесет вам ни одной строки, если в месяц не было продаж ни в одном году.
Насчет уценки - да, это расстраивает. Редактор сделал предварительный просмотр моей таблицы HTML, но после публикации она исчезла - поэтому пришлось удалить все форматирование HTML из сообщения ...
@kcrumley Я думаю, что мы пришли к аналогичным выводам. Этот запрос легко становится по-настоящему уродливым. Я действительно решил это, прежде чем читать ваш ответ, используя аналогичный (но все же другой подход). У меня есть доступ для создания хранимых процедур и функций в базе данных отчетов. Я создал функцию Table Valued, приняв в качестве параметра категорию продукта и финансовый год. Исходя из этого, функция заполнит таблицу, содержащую 12 строк. Строки будут заполнены данными из представления, если доступны какие-либо продажи, в противном случае строка будет иметь 0 значений.
Затем я присоединяюсь к двум таблицам, возвращаемым функциями. Поскольку я знаю, что на всех столах будет двенадцать ровностей, это будет проще, и я могу присоединиться к категории продуктов и месяцам:
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 довольно запутана! Но, по крайней мере, это работает - что меня пока радует;)
Case Statement - мой лучший друг по sql. Вам также понадобится таблица времени, чтобы сгенерировать 0 оборотов за оба месяца.
Предположения основаны на наличии следующих таблиц:
sales: Category | Revenue | Yearh | Month
и
tm: Year | Month (populated with all dates required for reporting)
Пример 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 имеет синтаксис сводной таблицы, который также будет делать это.
Вот несколько дополнительных ресурсов. ДЕЛО http://www.4guysfromrolla.com/webtech/102704-1.shtml SQL SERVER 2005 PIVOT http://msdn.microsoft.com/en-us/library/ms177410.aspx
Уловка состоит в том, чтобы выполнить FULL JOIN с 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 и Dynamic 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