Diskuze: MySQL - načtení řádků z tabulky do sloupců

Ostatní jazyky SQL SQL a databáze MySQL - načtení řádků z tabulky do sloupců

Avatar
Ladik777
Neregistrovaný
Avatar
Ladik777:

Ahoj,
prosím o radu, jak v MySQL načíst řádky tabulky do sloupců. Jednoduchý příklad:

CREATE TABLE zbozi (
id int unsigned NOT NULL AUTO_INCREMENT PRIMARY KEY
) ENGINE='MyISAM';

insert into zbozi values (1),(2),(3);

Tabulka Zbozi
id
--
1
2
3

CREATE TABLE ceny (
Zbozi_id int unsigned NOT NULL,
mena char(3) NOT NULL,
hodnota int unsigned NOT NULL
) ENGINE='MyISAM';

ALTER TABLE ceny
ADD PRIMARY KEY Zbozi_id_mena (Zbozi_id, mena),
ADD INDEX Zbozi_id (Zbozi_id);

insert into ceny values
(1, "kc", 100),
(1, "eur", 4),
(2, "kc", 200);

Tabulka Ceny
Zbozi_id | mena | hodnota


1 | kc | 100
1 | eur | 4
2 | kc | 200

a jako výsledek příkazu select potřebuji

id | kc | eur


1 | 100 | 4
2 | 200 | 0
3 | 0 | 0

Příklad je zjednodušený - v reálu tabulka Ceny jsou parametry zboží, kde každé zboží má různé parametry (v příkladu jako kc a eur) a různé hodnoty. Všechny parametry nejsou dopředu známé, tak na ně nelze připravit sloupce v tabulce.

Prozatím select mám následovně, ale předpokládám že by mohlo existovat lepší řešení.

select id,
ifnull((select hodnota from ceny where zbozi_id=Zbozi.id and mena="kc" limit 1),0) as kc,
ifnull((select hodnota from ceny where zbozi_id=Zbozi.id and mena="eur" limit 1),0) as eur
from zbozi order by id asc;

V PHP bych si toto vytvořil programově, ale rád bych nechal co nejvíce práce na MySQL. Děkuji za odpověď.

 
Odpovědět 23.5.2013 10:52
Avatar
Kit
Redaktor
Avatar
Odpovídá na Ladik777
Kit:

Udělej si zvlášť tabulku Kc a Eur. EAV není dobré používat, protože se s tím pak špatně pracuje.

SELECT id, Kc.cena AS kc, Eur.cena AS eur FROM zbozi
   LEFT JOIN Kc USING (id)
   LEFT JOIN Eur USING (id);
Nahoru Odpovědět  +1 23.5.2013 11:22
Vlastnosti objektů by neměly být veřejné. A to ani prostřednictvím getterů/setterů.
Avatar
Ladik777
Neregistrovaný
Avatar
Odpovídá na Kit
Ladik777:

No jo, ale když dopředu neznám počet parametrů a každé zboží může mít odlišné parametry, tak právě zakládat na vše tabulky mi nepřipadá vhodné (navíc kvůli právům by aplikace musela mít možnost tabulky zakládat). Ty ceny jsem uvedl jako velmi zjednodušený příklad principu co potřebuji z MySQL dostat.

S tim JOIN mě ještě napadlo:

select id, ifnull(kc.hodnota, 0) as kc, ifnull(eur.hodnota, 0) as eur
from zbozi left join ceny as kc on zbozi.id=kc.Zbo­zi_id and kc.mena="kc"
left join ceny as eur on zbozi.id=eur.Zbo­zi_id and eur.mena="eur"
order by id asc;

 
Nahoru Odpovědět 23.5.2013 11:50
Avatar
Kit
Redaktor
Avatar
Odpovídá na Ladik777
Kit:

Co vím, tak aplikace běžně mívají právo zakládat tabulky. Pochybuji, že ta práva nějak řešíš. Rozhodně je lepší za běhu přidat tabulku než alterovat stávající tabulku přidáváním dalšího sloupce.

Proč dáváš "0" místo null? To budou výrobky bez uvedené ceny zadarmo?

Editováno 23.5.2013 12:03
Nahoru Odpovědět 23.5.2013 12:00
Vlastnosti objektů by neměly být veřejné. A to ani prostřednictvím getterů/setterů.
Avatar
Ladik777
Neregistrovaný
Avatar
Odpovídá na Kit
Ladik777:

Práva webové aplikaci nastavuji v databázi co nejmenší možná.

Je jedno, jestli to vrátí NULL nebo 0, byl to jen příklad pro zjedodušení - reálná situace je, že to vůbec nejsou ceny, ale jsou to parametry zboží, takže tabulka je

Zbozi_id | parametr | hodnota


1 | délka | 100
1 | šířka | 15
2 | délka | 200

A když teď budu potřebovat přidat parametr třeba hmotnost, tak všude bude NULL, protože to není z čeho počítat (je jasné že měna v příkladu jde bez vyplnění hodnoty alespoň přibližně vypočítat převodem z jiné měny a zboží nebude zadarmo, ale hmost z délky a šířky spočítat nejde).

Potřeboval bych přesně to co je v původním dotazu. Našel jsem pár diskuzí v angličtině, kde se tímto zabývali, ale všechno mi přišlo dosti krkolomné a chci najít neoptimálnější postup, protože ho pak budu používat "na věky".

 
Nahoru Odpovědět 23.5.2013 12:20
Avatar
Kit
Redaktor
Avatar
Odpovídá na Ladik777
Kit:

EAV můžeš použít téměř stejným způsobem, jaký jsem popsal. Jen místo

LEFT JOIN Kc USING (id)

použiješ

LEFT JOIN (SELECT id, hodnota FROM cena WHERE mena="kc") AS Kc USING (id)

Prostě jde to také, jen je to o něco větší pakárna při skládání dotazu a je to o něco pomalejší.

Mnohem lépe by se to řešilo v PostgreSQL, kde je na to určen datový typ hstore. Pokud je to možné, přejdi raději na PostgreSQL. Budeš mít méně problémů, můžeš mít celý ceník v jediné tabulce a bude to i rychlejší.

Nahoru Odpovědět 23.5.2013 12:37
Vlastnosti objektů by neměly být veřejné. A to ani prostřednictvím getterů/setterů.
Avatar
Ladik777
Neregistrovaný
Avatar
Odpovídá na Kit
Ladik777:

Děkuji za informaci - v podstatě to nijak jednoduše v MySQL řešit nejde a to mi jako odpověď stačí.

 
Nahoru Odpovědět 23.5.2013 13:48
Avatar
Ladik777
Neregistrovaný
Avatar
Ladik777:

Možná nejlepší řešení, ale výkonem jsem to neporovnával na velkých datech.

select id, max(if(c.mena="kc",hod­nota,0)) as kc, max(if(c.mena="e­ur",hodnota,0)) as eur
from zbozi left join ceny as c on zbozi.id=c.Zbozi_id
group by id order by id asc;

 
Nahoru Odpovědět 23.5.2013 23:06
Děláme co je v našich silách, aby byly zdejší diskuze co nejkvalitnější. Proto do nich také mohou přispívat pouze registrovaní členové. Pro zapojení do diskuze se přihlas. Pokud ještě nemáš účet, zaregistruj se, je to zdarma.

Zobrazeno 8 zpráv z 8.