Как показать данные таблицы в виде столбцов в TSQL

У меня 2 таблицы в sql-server 2005 express DB

1. Таблица размеров предметов

CREATE TABLE [dbo].[ItemSize](
[ID] [int] IDENTITY(1,1) NOT NULL,
[iSize] [int] NULL
) ON [PRIMARY]

2. SalesLog Таблица

CREATE TABLE [dbo].[SalesLog](
[SalesID] [int] IDENTITY(1,1) NOT NULL,
[MemoNo] [int] NULL,
[ProductCode] [int] NULL,
[Quantity] [int] NULL,
[Price] [int] NULL,
[pGroup] [int] NULL,
[pName] [nvarchar](30) NULL,
[pSize] [int] NULL,
[BillDate] [nchar](10) NULL,
CONSTRAINT [PK_SalesLog] PRIMARY KEY CLUSTERED 
(
[SalesID] ASC
)WITH (PAD_INDEX  = OFF, STATISTICS_NORECOMPUTE  = OFF, IGNORE_DUP_KEY = OFF, 
ALLOW_ROW_LOCKS  = ON, ALLOW_PAGE_LOCKS  = ON) ON [PRIMARY]
) ON [PRIMARY]

Таблица ItemSize содержит различный размер напитка, например

100мл
200мл
300мл
400мл
500мл

Я хочу, чтобы данные столбцов таблицы ItemSize (iSize) были похожи на столбцы

ItemName   100ml    200ml    300ml   400ml   500ml
 Drink A    10       20        5       4      30
 Drink B    20       10        15      35     40 

и данные будут упорядочены из таблицы продаж в соответствии с размером и названием продукта, как показано выше.

У меня есть похожий запрос, но он не динамический. Я имею в виду, если я добавлю новый размер в таблицу ItemSize, то он не будет доступен для отчета. Поэтому мне нужно решение для этого.

В настоящее время я использую этот запрос для получения продаж в зависимости от размера товара

select i.gName,
isnull((select 
sum(Quantity)from saleslog where BillDate='01-06-2010' and pSize=0 and 
pGroup=i.gCode),0) as '0ml', 
isnull((select 
sum(Quantity)from saleslog where BillDate='01-06-2010' and pSize=180 and 
pGroup=i.gCode),0) as '180ml', 
isnull((select 
sum(Quantity)from saleslog where BillDate='01-06-2010' and pSize=375 and 
pGroup=i.gCode),0) as '375ml', 
isnull((select 
sum(Quantity)from saleslog where BillDate='01-06-2010' and pSize=500 and 
pGroup=i.gCode),0) as '500ml',
isnull((select 
sum(Quantity)from saleslog where BillDate='01-06-2010' and pSize=650 and 
pGroup=i.gCode),0) as '650ml',
isnull((select 
sum(Quantity)from saleslog where BillDate='01-06-2010' and pSize=750 and 
pGroup=i.gCode),0) as '750ml',
isnull((select 
sum(Quantity)from saleslog where BillDate='01-06-2010' and pSize=1000 and 
pGroup=i.gCode),0) as '1000ml',
isnull((select 
sum(Quantity)from saleslog where BillDate='01-06-2010' and 
pSize=2000 and pGroup=i.gCode),0) as '2000ml' 
from saleslog as s 
inner join ItemGroup as i on s.pGroup=i.gCode 
where BillDate='01-06-2010' 
group by i.gCode, i.gName

1 ответ

Решение

Вы проверили функцию PIVOT? Вы не упоминаете, какую версию SQL вы используете, но она доступна в 2008 году и делает то, что вам нужно.

Кроме того, ваш код кажется ужасно неэффективным. Вы делаете выбор внутри, выбирает данные, к которым у вас уже есть доступ. Может быть, мне трудно отождествлять себя с вашим набором данных, но кажется, что в вашей текущей ситуации вам нужна инструкция case, а не множество вариантов выбора. Я считаю, что следующие работы:

select
  i.gName,
  case i.pSize when 0 then sum(Quantity) else 0 end as [0ml],
  case i.pSize when 180 then sum(Quantity) else 0 end as [180ml],
  case i.pSize when 375 then sum(Quantity) else 0 end as [375ml],
  { rest of case samples }
from
  saleslog as s
    inner join ItemGroup as i on s.pGroup = i.gGroup
where
  s.BillDate = '01-06-2010'
group by
  i.gName

Я также немного запутался в вашем вопросе. Вы упомянули, что у вас есть две таблицы, одна имеет идентификатор и iSize, но я не вижу iSize, поэтому я предполагаю, что pSize(?). Кроме того, ваша таблица продаж не совпадает с вашим запросом, т. Е. В вашей таблице продаж нет количественного количества, в вашем создании упоминается [продажи], но ваш фактический запрос использует [продажи данных].

Может быть, еще уточнения?

Другие вопросы по тегам