2008-08-19 3 views
4

У меня есть вид базы данных Monthly Status Мне нужно создать отчет на основе. Данные в представлении выглядит примерно так:SQL-запрос для сравнения продаж продукта по месяцам

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 для этого запроса, поэтому, если есть какие-то полезные новые функции, которые могут помочь мне, дайте мне знать.

ответ

4

Деловая заявка - мой лучший друг в 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 имеет поворотный синтаксис, который будет делать это, а также.

Вот некоторые дополнительные ресурсы. СЛУЧАЯ http://www.4guysfromrolla.com/webtech/102704-1.shtml SQL SERVER 2005 PIVOT http://msdn.microsoft.com/en-us/library/ms177410.aspx

0

Возможно, я ошибаюсь, но разве вы не должны использовать полное внешнее соединение вместо левого соединения? Таким образом вы получите «пустые» столбцы из обеих таблиц.

http://en.wikipedia.org/wiki/Join_(SQL)#Full_outer_join

3

@Christian - уценки редактор - UGH; особенно если предварительный просмотр и окончательная версия вашего сообщения не согласуются ... @Christian - полное внешнее соединение - полное внешнее соединение отменено тем фактом, что в отношении WHERE есть ссылки на SP1, а предложение WHERE - применяется после СОЕДИНЕНИЯ. Чтобы сделать полное внешнее соединение с фильтрацией в одной из таблиц, вам нужно поместить предложение WHERE в подзапрос, поэтому фильтрация происходит до соединения или пытайтесь построить все ваши критерии WHERE в предложении JOIN ON, который безумно уродлив. Ну, на самом деле нет такого красивого способа сделать это.

@Jonas: Учитывая это:

Кроме того, в докладе фактически за финансовый год - так я хотел бы иметь пустые столбцы с 0 в обоих, если не было продаж, скажем, месяц 5 для любой 2007 или 2008.

и тот факт, что эта работа не может быть выполнена с помощью довольно запроса, я определенно попытаюсь получить результаты, которые вы действительно хотите. Нет смысла иметь уродливый запрос и даже не получать точные данные, которые вы действительно хотите.;)

Итак, я предлагаю сделать это в 5 шагов:
1. создать временную таблицу в формате, который вы хотите, чтобы ваши результаты, чтобы соответствовать
2. заполнит его с двенадцатью рядами, с 1-12 в месяц колонка
3. обновление столбца «в этом году» с помощью SP1 логики
4. обновление в колонке «Последний год» с помощью SP2 логики
5. выберите из временной таблицы

конечно, я Думаю, я работаю из предположения, что вы можете создать хранимую процедуру для этого. Вы можете технически иметь возможность запускать весь этот пакет, но такое уродство очень редко встречается. Если вы не можете сделать SP, я предлагаю вам вернуться к полному внешнему соединению через подзапрос, но он не даст вам строки, если в месяц не было продаж ни в год.

1

Об уценке - Да, это расстраивает. Редактор сделал предварительный просмотр моей таблицы 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 довольно грязная! Но, по крайней мере, это работает - что делает меня счастливым пока;)

1

Хитрость заключается в том, чтобы сделать полный 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 годах или и то, и другое. Чтобы получить строки без продаж в год, вам просто нужно ввести INNER JOIN в таблицу с номерами 1-12 (вы делаете have one of those, не так ли?).

0

С помощью поворота и 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 ('[email protected]+') 
      ) 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 
Смежные вопросы