Vianoce v ITnetwork sú tu! Dobí si teraz kredity a získaj až 80 % extra kreditov na e-learningové kurzy ZADARMO. Zisti viac.
Hľadáme nové posily do ITnetwork tímu. Pozri sa na voľné pozície a pridaj sa k najagilnejšej firme na trhu - Viac informácií.

10. diel - MS-SQL krok za krokom: Ďalšie dotazy a väzba M:N

V minulej lekcii, MS-SQL krok za krokom: Dotazy cez viac tabuliek (JOIN), sme sa naučili JOINy, teda dotazy nad viacerými tabuľkami.

V tomto duchu budeme dnes v MS-SQL tutoriále pokračovať, založíme si ešte jednu tabuľku s komentármi a potom si povieme niečo o väzbe M:N.

Tabuľka komentáre

Pokračujme v našom redakčnom systéme a vytvorme si tabuľku Comments. Komentár sa (podobne ako článok) viaže na používateľa. Viaže sa ale aj na článok. Máme tu teda dve väzby 1:N. Jeden článok má N komentárov, jeden používateľ má N komentárov. Komentár patrí vždy iba jednému užívateľovi a jednému článku.

Keďže komentár je časť a patrí do dvoch celkov (k článku ak užívateľovi), bude obsahovať dva stĺpce s Id článku a Id užívateľa. Týmto stĺpcom s Id položky z cudzej tabuľky hovoríme cudzie kľúče. Už ich poznáme z minula (pri článku bol cudzí kľúč užívateľa), len sme si nepovedali, že sa im tak hovorí. Okrem nich bude mať komentár text a dátum:

CREATE TABLE [Comments] (
    [Id] INT IDENTITY,
    [ArticleId] INT,
    [UserId] INT,
    [Content] NVARCHAR(MAX),
    [Date] DATETIME,
    PRIMARY KEY ([Id])
);

Vložme si nejaké komentáre:

INSERT INTO [Comments] ([UserId], [Content], [Date], [ArticleId]) VALUES
(4, 'Great article!', '2012-4-6', 1),
(4, 'So what exactly is the condition for the formation of bacteria?', '2011-1-28', 2),
(1, 'I am stuck in this game, where do I find the key to the 3rd level?', '2011-9-30', 3),
(4, 'How do I make the platform move in the 5th level?', '2010-8-1', 3),
(1, 'I died and I did not save the game, what should I do?', '2012-4-14', 4),
(3, 'Good game!', '2012-4-6', 4),
(3, 'I do not understand it!', '2011-4-6', 1),
(2, 'Great article!', '2012-5-6', 1);

Poďme si skúsiť vypísať všetky komentáre spolu s ich autormi a článkami, ku ktorým patria. JOINy už vieme, tento dotaz bude obdobný, len bude rovno cez dve tabuľky naraz, čiže s dvoma JOINmi. JOINov môžeme mať v dotaze samozrejme koľko chceme, ale mali by sme pamätať na to, že to nie sú pre databázu úplne jednoduché operácie:

SELECT [Users].[Nickname], [Comments].[Content], [Articles].[Title]
FROM [Comments]
INNER JOIN [Users] ON [Users].[Id] = [Comments].[UserId]
INNER JOIN [Articles] ON [Articles].[Id] = [Comments].[ArticleId]
ORDER BY [Comments].[Date];

Výsledok:

Nickname Content Title
Emma How do I make the platform move in the 5th level? Cheese Mouse
Emma So what exactly is the condition for the formation of bacteria? Bacteria
Denny I do not understand it! Algorithm
Michael I am stuck in this game, where do I find the key to the 3rd level? Cheese Mouse
Emma Great article! Algorithm
Denny Good game! Pacman
Michael I died and I did not save the game, what should I do? Pacman
David Great article! Algorithm

Všimnite si, že sme všetky stĺpce predsadili názvom tabuľky. Malo by sa to tak robiť vždy. Tu konkrétne sa volá Content obsah komentára aj obsah článku. Pri zložitejšej štruktúre databázy sa toto stáva so stĺpcami ako Date, Id, Author...

Skúste si dotaz bez názvov tabuliek, nebude fungovať:

-- tento dotaz nebude fungovat
SELECT [Users].[Nickname], [Comments].[Content], [Articles].[Title]
FROM [Comments]
INNER JOIN [Users] ON [Id] = [UserId]
INNER JOIN [Articles] ON [Id] = [ArticleId]
ORDER BY [Comments].[Date];

MS-SQL vyhodí hlášku:

Msg 209, Level 16, State 1, Line 4
Ambiguous column name 'Id'.
Msg 209, Level 16, State 1, Line 5
Ambiguous column name 'Id'.

Pri zložitejších dotazoch cez viac tabuliek môže byť výhodné použiť aliasy. Aliasy už tiež vieme, deklarujú sa cez kľúčové slovo AS. Použime ich v tomto dotazu:

SELECT [U].[Nickname], [C].[Content], [A].[Title]
FROM [Comments] AS [C]
INNER JOIN [Users] AS [U] ON [U].[Id] = [C].[UserId]
INNER JOIN [Articles] AS [A] ON [A].[Id] = [C].[ArticleId]
ORDER BY [C].[Date];

Dotaz vyzerá oveľa prehľadnejšie, nemusíme opisovať názvy tabuliek. Skrátili sme si ich, tu len na počiatočné písmená.

Sekcie

Pokračujme v štruktúre redakčného systému. Články sa radia do sekcií, tie sú uložené v tabuľke Sections. Je tu však malý háčik. Jedna sekcia môže obsahovať niekoľko článkov. Jeden článok však môže tiež patriť do niekoľkých sekcií.

Na účely redakčného systému by samozrejme stačilo, aby článok spadal vždy len do jednej sekcie. Tak by sme sa ale nič nenaučili :)

Narážame na väzbu M:N.

Väzba M:N

Väzbu M:N sme si už vysvetlili, ďalším príkladom by mohli byť napríklad študent a predmet. Každý študent chodí na niekoľko predmetov a každý predmet má niekoľko študentov, ktorí naň dochádza.

Poďme si založiť tabuľku sekcií. Bude veľmi triviálna, pretože v nej budú len dva stĺpce. Jeden s Id sekcie a druhý s jej názvom.

CREATE TABLE [Sections] (
    [Id] INT IDENTITY,
    [Name] NVARCHAR(155),
    PRIMARY KEY ([Id])
);

Naplňme si ju dátami:

INSERT INTO [Sections] ([Name]) VALUES
('Algorithms'),
('Games');

Databáza ako taká väzbu M:N nevie. To pre nás ale nie je prekážkou a bežne sa to obchádza vytvorením tzv. väzobnej tabuľky. Väzobná tabuľka nenesie sama o sebe žiadne dáta a slúži iba na prepojenie dvoch tabuliek. Každý riadok väzobnej tabuľky bude obsahovať Id článku a Id sekcie, tak ich spolu prepoja. Vďaka tomu môžeme dotazom zistiť, aké články sú v sekcii alebo do ktorých sekcií článok patrí. Založme si väzobnú tabuľku, pomenujeme ju ArticleSection:

CREATE TABLE [ArticleSection] (
    [Id] INT IDENTITY,
    [ArticleId] INT,
    [SectionId] INT,
    PRIMARY KEY ([Id])
);

Teraz ju naplníme dátami, ktoré nám články a sekcie prepoja:

INSERT INTO [ArticleSection] ([ArticleId], [SectionId]) VALUES
(1, 1),
(2, 1),
(2, 2),
(3, 2),
(4, 2);

A skúsme si dalšie dotaz. Vypíšme si články v sekcii Algoritms. Vyberieme články, tie prepojíme pomocou tabuľky ArticleSection so sekciou:

SELECT [A].[Url], [A].[Title]
FROM [Articles] AS [A]
INNER JOIN [ArticleSection] AS [AS] ON [AS].[ArticleId] = [A].[Id]
INNER JOIN [Sections] AS [S] ON [AS].[SectionId] = [S].[Id]
WHERE [S].[Name] = 'Algorithms';

Výsledok:

Url Title
what-is-an-algorithm Algorithm
bacteria-cell-automatic-machine Bacteria

Dotaz vyššie by bol na webe naozaj použitý na vypísanie obsahu sekcie. Podľa väzobnej tabuľky sme prepojili články so sekciou. Vlastne sme pripojili tie riadky väzobnej tabuľky, ktoré spájajú daný článok a ten článok potom k jeho sekcii.

To by dnes už stačilo. Ono by to vôbec na chvíľu stačilo, už toho vieme dosť.

V nasledujúcom cvičení, Riešené úlohy k 8.-10. lekciu MS-SQL, si precvičíme nadobudnuté skúsenosti z predchádzajúcich lekcií.


 

Predchádzajúci článok
MS-SQL krok za krokom: Dotazy cez viac tabuliek (JOIN)
Všetky články v sekcii
MS-SQL databázy krok za krokom
Preskočiť článok
(neodporúčame)
Riešené úlohy k 8.-10. lekciu MS-SQL
Článok pre vás napísal Michal Žůrek - misaz
Avatar
Užívateľské hodnotenie:
4 hlasov
Autor se věnuje tvorbě aplikací pro počítače, mobilní telefony, mikroprocesory a tvorbě webových stránek a webových aplikací. Nejraději programuje ve Visual Basicu a TypeScript. Ovládá HTML, CSS, JavaScript, TypeScript, C# a Visual Basic.
Aktivity