SQL транспонировать строки в виде столбцов - PullRequest
37 голосов
/ 20 января 2010

У меня есть интересная головоломка, которую, я считаю, можно решить исключительно на SQL.У меня есть таблицы, похожие на следующие:

responses:

user_id | question_id | body
----------------------------
1       | 1           | Yes
2       | 1           | Yes
1       | 2           | Yes
2       | 2           | No
1       | 3           | No
2       | 3           | No


questions:

id | body
-------------------------
1 | Do you like apples?
2 | Do you like oranges?
3 | Do you like carrots?

, и я хотел бы получить следующий вывод

user_id | Do you like apples? | Do you like oranges? | Do you like carrots?
---------------------------------------------------------------------------
1       | Yes                 | Yes                  | No
2       | Yes                 | No                   | No

Я не знаю, сколько будет вопросов, и они будутбыть динамичным, поэтому я не могу просто написать код для каждого вопроса.Я использую PostgreSQL и считаю, что это называется транспонированием, но я не могу найти ничего, что говорит о стандартном способе сделать это в SQL.Я помню, как делал это в своем классе баз данных еще в колледже, но это было в MySQL, и я, честно говоря, не помню, как мы это делали.

Я предполагаю, что это будет комбинация объединений и GROUP BY утверждение, но я даже не могу понять, с чего начать.

Кто-нибудь знает, как это сделать?Большое спасибо!

Редактировать 1: Я нашел некоторую информацию об использовании кросс-таблицы , которая, кажется, то, что я хочу, но у меня возникают проблемы с пониманиемЭто.Ссылки на лучшие статьи будут с благодарностью!

Ответы [ 5 ]

48 голосов
/ 20 января 2010

Использование:

  SELECT r.user_id,
         MAX(CASE WHEN r.question_id = 1 THEN r.body ELSE NULL END) AS "Do you like apples?",
         MAX(CASE WHEN r.question_id = 2 THEN r.body ELSE NULL END) AS "Do you like oranges?",
         MAX(CASE WHEN r.question_id = 3 THEN r.body ELSE NULL END) AS "Do you like carrots?"
    FROM RESPONSES r
    JOIN QUESTIONS q ON q.id = r.question_id
GROUP BY r.user_id

Это стандартный сводный запрос, поскольку вы «поворачиваете» данные из строк в столбчатые данные.

11 голосов
/ 19 октября 2015

Я реализовал по-настоящему динамическую функцию для решения этой проблемы без необходимости жесткого кода какого-либо конкретного класса ответов или использования внешних модулей / расширений.Он также дает полный контроль над порядком столбцов и поддерживает несколько столбцов ключей и классов / атрибутов.

Вы можете найти его здесь: https://github.com/jumpstarter-io/colpivot

Пример, решающий эту конкретную проблему:

begin;

create temporary table responses (
    user_id integer,
    question_id integer,
    body text
) on commit drop;

create temporary table questions (
    id integer,
    body text
) on commit drop;

insert into responses values (1,1,'Yes'), (2,1,'Yes'), (1,2,'Yes'), (2,2,'No'), (1,3,'No'), (2,3,'No');
insert into questions values (1, 'Do you like apples?'), (2, 'Do you like oranges?'), (3, 'Do you like carrots?');

select colpivot('_output', $$
    select r.user_id, q.body q, r.body a from responses r
        join questions q on q.id = r.question_id
$$, array['user_id'], array['q'], '#.a', null);

select * from _output;

rollback;

Это выводит:

 user_id | 'Do you like apples?' | 'Do you like carrots?' | 'Do you like oranges?' 
---------+-----------------------+------------------------+------------------------
       1 | Yes                   | No                     | Yes
       2 | Yes                   | No                     | No
6 голосов
/ 21 ноября 2012

Этот пример можно решить с помощью функции crosstab следующим образом

drop table if exists responses;
create table responses (
user_id integer,
question_id integer,
body text
);

drop table if exists questions;
create table questions (
id integer,
body text
);

insert into responses values (1,1,'Yes'), (2,1,'Yes'), (1,2,'Yes'), (2,2,'No'), (1,3,'No'), (2,3,'No');
insert into questions values (1, 'Do you like apples?'), (2, 'Do you like oranges?'), (3, 'Do you like carrots?');

select * from crosstab('select responses.user_id, questions.body, responses.body from responses, questions where questions.id = responses.question_id order by user_id') as ct(userid integer, "Do you like apples?" text, "Do you like oranges?" text, "Do you like carrots?" text);

Сначала необходимо установить расширение tablefunc.Начиная с версии 9.1, вы можете сделать это, используя расширение create:

CREATE EXTENSION tablefunc;
2 голосов
/ 26 сентября 2014

Я написал функцию для генерации динамического запроса.Он генерирует sql для кросс-таблицы и создает представление (сначала удаляет его, если оно существует).Вы можете выбрать из вида, чтобы получить результаты.

Вот функция:

CREATE OR REPLACE FUNCTION public.c_crosstab (
  eavsql_inarg varchar,
  resview varchar,
  rowid varchar,
  colid varchar,
  val varchar,
  agr varchar
)
RETURNS void AS
$body$
DECLARE
    casesql varchar;
    dynsql varchar;    
    r record;
BEGIN   
 dynsql='';

 for r in 
      select * from pg_views where lower(viewname) = lower(resview)
  loop
      execute 'DROP VIEW ' || resview;
  end loop;   

 casesql='SELECT DISTINCT ' || colid || ' AS v from (' || eavsql_inarg || ') eav ORDER BY ' || colid;
 FOR r IN EXECUTE casesql Loop
    dynsql = dynsql || ', ' || agr || '(CASE WHEN ' || colid || '=''' || r.v || ''' THEN ' || val || ' ELSE NULL END) AS ' || agr || '_' || r.v;
 END LOOP;
 dynsql = 'CREATE VIEW ' || resview || ' AS SELECT ' || rowid || dynsql || ' from (' || eavsql_inarg || ') eav GROUP BY ' || rowid;
 RAISE NOTICE 'dynsql %1', dynsql; 
 EXECUTE dynsql;
END

$body$
LANGUAGE 'plpgsql'
VOLATILE
CALLED ON NULL INPUT
SECURITY INVOKER
COST 100;

А вот как я ее использую:

SELECT c_crosstab('query_txt', 'view_name', 'entity_column_name', 'attribute_column_name', 'value_column_name', 'first');

Пример: Вы запускаете кулак:

SELECT c_crosstab('Select * from table', 'ct_view', 'usr_id', 'question_id', 'response_value', 'first');

Чем:

Select * from ct_view;
0 голосов
/ 20 января 2010

Пример этого есть в contrib/tablefunc/.

...