|
|||||||||||||||||||||||||||||||||||||||||||||||||
Menu
Distributions (131)
bootable [55]
commercial [7] no-commercial [42] unclassified [20] [7]
Software (10844)
|
MySQL (28) - Dotazy pro pokročiléDnes o tom, jak vrátit sadu záznamů s očíslovanými řádky. Uvidíte, jak se dá vtipně využít spojení tabulky se sebou samou.
Minule jsme načali látku o pokročilých technikách sestavování SQL
dotazů tím, že jsme si ukázali, jak se dá sestavit dotaz na duplicitní
položky a dotaz na chybějící záznamy. Dnes budeme v rozboru
náročnějších dotazů pokračovat. Ukážeme si, co může vzniknout za
problémy, když se rozhodnete vracet pořadová čísla záznamů. "Očíslované" záznamyČas od času je třeba vrátit sadu záznamů, která obsahuje jako součást vracených dat i pořadové číslo záznamu. Může to být potřeba například při sestavování výsledkových listin v nějakém soutěžním klání, jako podklad pro tvorbu kalendářů a podobně. Mějme třeba jako data tabulku: create table zamestnanci
(jmeno varchar(20), plat integer);
a chtějme vrátit lidi podle jejich platů (nejbohatší nejvýše) s tím,
že seznam bude očíslován. Jak na to? Existuje několik řešení. Řešení s dočasnou tabulkouPrvní, co by někoho mohlo napadnout je, že k tabulce by se mohlo přidat automaticky číslované pole, a pak ji prostě vrátit i s tímto polem: alter table zamestnanci
add id int not null auto_increment primary key;
Tento přístup má však mnoho omezení, v praxi většinou takových, že
se to nedá použít vůbec:
Ačkoli by se zdálo, že se tím nemá cenu déle zabývat, může mít tato
technika vracení "očíslovaných záznamů" své zastánce, zejména pokud ji
trochu upravíme. MySQL totiž může fungovat takto:
Celé to předveďme na příkladu. Budeme chtít vrátit seznam zaměstnanců s tím, že budou seřazeni podle platu (sestupně) a očíslováni. Řešení? create temporary table
tmp_zamestnanci like zamestnanci;
Toto řešení samozřejmě předpokládá, že v původní tabulce
neexistovalo pole id. Pokud by existovalo, je druhý řádek z příkazu
výše zbytečný. Tahle metoda je občas cenná, protože na jednu stranu
sice přesypává kvanta čísel, na stranu druhou však může i tak být dost
rychlá. Druhé řešení, které vám předvedu, totiž trpí poměrně výžnými
výkonostními problémy. Řešení s poddotazemTohle řešení nepoužívá dočasnou tabulku, ale poddotaz. A to poddotaz korelační, tedy takový, kde se data z vnitřního dotazu vztahují k dotazu vnějšímu. Nejprve jej předvedu: select zamestnanci.*,
(select count(*) from zamestnanci as pocitadlo where pocitadlo.plat
>=zamestnanci.plat) as poradi from zamestnanci order by plat desc;
a teď vysvětlím - vnořený poddotaz prostě vrátí počet všech
zaměstnanců, kteří mají stejný nebo vyšší plat než aktuálně
zpracovávaný zaměstnanec. Což znamená, že de facto vrátí počet
zaměstnanců majících vyšší plat než zpracovávaný záznam. Jak už to na
tom světě bývá, i tohle řešení má háček, a to ten, že když by měli dva
pracovníci stejný plat, vrátí jim poddotaz stejné pořadí. Což je
logické - v takovém případě totiž existuje "stejný počet kolegů,
majících stejný či vyšší plat". Jak z toho ven? Je tu jeden způsob - a to ten, že si "vypůjčíme" ještě nějaké další pole pro řazení. V našem případě bychom například mohli řadit jednak sestupně podle platu, a jednak jména podle abecedy. To všechno proto, abychom měli zajištěno pevné pořadí i v rámci skupiny se stejným platem. Dotaz, fungující dobře i pro záznamy s více lidmi se stejným platem by pak mohl vypadat takto (důležité pasáže jsem zvýraznil): select zamestnanci.*,
(select count(*) from zamestnanci as pocitadlo where pocitadlo.plat
>zamestnanci.plat or (pocitadlo.plat=zamestnanci.plat
and
pocitadlo.jmeno<=zamestnanci.jmeno)) as poradi from
zamestnanci
order by plat desc, jmeno;
Když si poddotaz přečtete a zamyslíte se nad ním, je to opět
logické. Vrací totiž:
Pochopitelně, poddotaz by nefungoval správně, kdyby řazení v hlavním
dotazu bylo jiné. Pozn.: Vnímavý čtenář může
namítnout, že v reále lze mít dva lidi
jak se stejným jménem, tak i se stejným platem. Ono se to ve
skutečnosti většinou ještě řadí podle nějakého jedinečného
identifikátoru řádku (například, podle automaticky číslovaného pole),
takže tento problém nenastává. Řešení se spojenímPokud jste se rozhodli přepsat výše uvedený poddotaz na spojení, mám pro vás dobrou zprávu - jde to. Člověk musí pouze dát pozor na to, že v hlavním dotazu je pak nutné použít klauzuli GROUP BY, neb agregační funkce COUNT(*) musí počítat pouze s počty "předcházejících" zaměstnanců pro konkrétního pracovníka. Mohlo by to tedy dopadnout nějak takto: select zamestnanci.*,
count(*) from zamestnanci join zamestnanci as pocitadlo on
pocitadlo.plat > zamestnanci.plat or (pocitadlo.plat =
zamestnanci.plat and pocitadlo.jmeno <= zamestnanci.jmeno) group by zamestnanci.jmeno,
zamestnanci.plat order by zamestnanci.plat desc,
zamestnanci.jmeno;
ZávěrJak vidíte, vracet jako součást dat i pořadová čísla řádků není až
tak úplně jednoduché. Někteří programátoři proto volí jiný přístup -
nechají data sestavit bez čísel řádků a ve volající aplikaci pak
zpracovávají data po jednotlivých záznamech. Aplikace sama si vytvoří a
spravuje jakýsi čítač, s nímž se pak manipuluje. Řešení, které jsem
předvedl, však může být mnohem elegantnější, protože udržuje aplikační
logiku v databázi a nemá ji roztříštěnu mezi databázi a aplikaci. Co si vyberete záleží pochopitelně na konkrétní situaci. Pokud budete číslovat větší počet záznamů, udělejte si nějaké testy rychlosti - jednotlivé metody zde uvedené se mohou mít až řádově odlišný čas provádění!
Related article
MySQL (1) - pestrý svět databází MySQL (2) - Instalujeme databázi MySQL MYSQL (3) Instalujeme MySQL podruhé MySQL (4) - něco terminologie MySQL (5) - tajuplné SQL MySQL (6) - Ukládáme řetězce MySQL (7) - hrátky s čísly MySQL (8) - Ukládání datumů MySQL (9) - Další datové typy MySQL (10) - tvorba databáze. Základy DDL MySQL (11) - vytváříme tabulky MySQL (12) - tipy k tvorbě tabulek MySQL (13) - Vkládáme data MySQL (14) - Upravujeme data MySQL (15) - Odstraňujeme data MySQL (16) - Tipy a triky k manipulaci s daty MySQL (17) - vybíráme data MySQL (18) - Filtrujeme data MySQL (19) - Řadíme data MySQL (20) - spojení více tabulek MySQL (21) - klauzule JOIN MySQL (22) - tipy a triky ke spojování tabulek MySQL (23) - relace 1:N a N:N MySQL (24) - Seskupujeme záznamy MySQL (25) - hrátky se seskupenými záznamy MySQL (26) - Poddotazy MySQL (27) - Složitější dotazy MySQL (29) - Vracení nejvyšších záznamů MySQL (30) - průběžné součty MySQL (31) - Indexy MySQL (32) - ještě k indexům MySQL (33) - Příkaz UNION MySQL (34) - větvení kódu a pivotní tabulky MySQL (35) - vestavěné funkce MySQL (36) - Regulární výrazy MySQL (37) - použití fulltextového vyhledávání MySQL (38) - Fulltext a praxe MySQL (39) - typy tabulek v MySQL MySQL (40) - další typy tabulek MySQL (41) - Transakce MySQL (42) - ještě k transakcím MySQL (43) - Uložené procedury MySQL (44) - parametry uložených procedur MySQL (45) - větvení kódu uložených procedur MySQL (46) - Triggery MySQL (47) - Triggery a praxe MySQL (48) - UDF MySQL (49) - pohledy MySQL (50) - Pohledy podruhé MySQL (51) - Metadata MySQL (52) - A co zálohování? MySQL (53) - SELECT INTO OUTFILE MySQL (54) - zálohování MySQL z webu MySQL (55) - zálohování MySQL z pohledu správce MySQL (56) - Obnova zálohovaných dat MySQL (57) - Ach, ta čeština MySQL (58) - čeština v praxi MySQL (59) - české řazení MySQL (60) - řádkový klient MySQL (61) - Oprávnění MySQL (62) - Oprávnění podruhé MySQL (63) - jemné nastavení práv MySQL (64) - nad dotazy čtenářů MySQL (65) - Ladíme server MySQL (66) - Ještě k ladění serveru MySQL - (67) MySQL (68) - Závěr MySQL (69) - Prepared Statements Previous Show category (serial) Next
|
Szukanie oprogramowania
|
|||||||||||||||||||||||||||||||||||||||||||||||
©Pavel Kysilka - 2003-2024 | maillinuxsoft.cz | Design: www.megadesign.cz |