Diskuze: Filtrace dostupných možností dle zvolených kritérií
V předchozím kvízu, Online test znalostí SQL a databází, jsme si ověřili nabyté zkušenosti z kurzu.

Člen

Zobrazeno 7 zpráv z 7.
//= Settings::TRACKING_CODE_B ?> //= Settings::TRACKING_CODE ?>
V předchozím kvízu, Online test znalostí SQL a databází, jsme si ověřili nabyté zkušenosti z kurzu.
S EF nepracuju, ale když nikdo neodpovídá, zkusím pomoct aspoň s klasickým SQL.
Nejsem si jistý, jak to myslíš s tím jedním dotazem. Podle mě potřebuješ pro naplnění každého comba samostatný select, který bude vypadat nějak takhle:
SELECT PW.Id, PW.[Description], COUNT(P.Id) AS [Count]
FROM PneuWidth PW
LEFT OUTER JOIN Pneu P ON P.PneuWidthId = PW.Id
AND P.PneuDiameterId = ISNULL(@PneuDiameterId, P.PneuDiameterId)
AND P.PneuProfileId = ISNULL(@PneuProfileId, P.PneuProfileId)
AND P.PneuSpeedIndexId = ISNULL(@PneuSpeedIndexId, P.PneuSpeedIndexId)
-- atd
GROUP BY PW.Id, PW.[Description]
HAVING COUNT(P.Id) > 0 -- tenhle radek tam dej jen kdyz chces vynechat nedostupne moznosti
ORDER BY PW.[Description]
Když uživatel něco vybere, pro každé combo znovu zavoláš jeho select a do parametrů @PneuDiameterId apod. vyplníš to, co uživatel vybral v ostatních combech. Kde nic nevybral, tam bude NULL, čímž se příslušná podmínka neaplikuje.
Celkový počet dostupných produktů bych si vytáhl samostatným jednoduchým selectem:
SELECT COUNT(1) AS TotalCount
FROM Pneu P
WHERE P.PneuDiameterId = ISNULL(@PneuDiameterId, P.PneuDiameterId)
AND P.PneuProfileId = ISNULL(@PneuProfileId, P.PneuProfileId)
AND P.PneuSpeedIndexId = ISNULL(@PneuSpeedIndexId, P.PneuSpeedIndexId)
-- ... atd, stejne jako vys
Pokud předem víš, že nebudeš chtít ukazovat kategorie, kde zbylo 0 produktů, dá se to udělat buď tím HAVING výše, nebo takhle, což by snad mělo být trochu rychlejší:
SELECT PW.Id, PW.[Description], COUNT(1) AS [Count]
FROM PneuWidth PW
LEFT OUTER JOIN Pneu P ON P.PneuWidthId = PW.Id
WHERE P.PneuDiameterId = ISNULL(@PneuDiameterId, P.PneuDiameterId)
AND P.PneuProfileId = ISNULL(@PneuProfileId, P.PneuProfileId)
AND P.PneuSpeedIndexId = ISNULL(@PneuSpeedIndexId, P.PneuSpeedIndexId)
-- atd
GROUP BY PW.Id, PW.[Description]
ORDER BY PW.[Description]
Na obrázcích máš dummy ukázku s filtrem a bez filtru.
Děkuji moc. Zkusím se od tohoto odpíchnout
Takže pro každou vlastnost ve filtraci, musím mít jeden select dotaz? To znamená, že Alza jich má desítky? Jelikož jejich filtrace obsahuje nespočet parametrů.
Pro doplnění. Já tabulku mám jen jednu a to Products se všemi vlastnostmi pneumatik atd. O co mi od začátku jde, jak vytáhnout z ní všechny možnosti pro selectBoxy co nejefektivněji. Nejsem v tom moc znalý, předpokládal jsem, že to půjde vytáhnout jedním dotazem, na to se tady ptám. Děkuji
Pokud mas 10 selectboxu, tak mas 10 sql dotazu.
Pokud je v tom problem, muzes to udelat jednim s UNION. Pro upravy a
jednoduchost je ale lepsi mit kazdy ciselnik (staty, okresy, mesta a pod) mit
zvlast. Samozrejme, je to o neco pomalejsi a vadi to tam, kde mas limit na pocet
sql dotazu za hodinu. Tak, na takovem hostingu nic neprovozuji
SELECT id, value, 0 AS type FROM tabulka1
UNION ALL
SELECT id, value, 1 AS type FROM tabulka2
UNION ALL
SELECT id, value, 2 AS type FROM tabulka3
U unionu je ale podstatne, ze musis mit stejny pocet sloupcu.
Promiň za zdržení v odpovědi, měl jsem toho hodně v práci. Samozřejmě to do jednoho výsledku spojit můžeš, pak bude mít asi takovouhle strukturu:
Combo | Moznost | NovyPocet |
---|---|---|
PneuSpeedIndex | T | 2 |
PneuSpeedIndex | U | 1 |
PneuWidth | 205 | 3 |
PneuWidth | 215 | 2 |
... | ... | ... |
Dotaz pak může vypadat asi takhle:
WITH P AS
(
SELECT P.Id, P.PneuWidth, P.PneuDiameter, P.PneuProfile, P.PneuSpeedIndex
FROM Products P
WHERE P.PneuWidth = ISNULL(@PneuWidth, P.PneuWidth)
AND P.PneuDiameter = ISNULL(@PneuDiameter, P.PneuDiameter)
AND P.PneuProfile = ISNULL(@PneuProfile, P.PneuProfile)
AND P.PneuSpeedIndex = ISNULL(@PneuSpeedIndex, P.PneuSpeedIndex)
)
SELECT 'PneuWidth' AS Combo, P.PneuWidth AS [Option], COUNT(P.Id) AS [Count]
FROM P
GROUP BY P.PneuWidth
UNION ALL
SELECT 'PneuDiameter' AS Combo, P.PneuDiameter AS [Option], COUNT(P.Id) AS [Count]
FROM P
GROUP BY P.PneuDiameter
UNION ALL
...
Fakticky to je pořád tolik selectů, kolik máš comboboxů, jinak se to
nedá udělat, protože každý groupuje podle něčeho jiného. Ten union je
jen takový fejk, jak to spojit do jednoho recordsetu. Jak píše Peter Mlich,
dej si pozor, že všechny selecty v unionu musejí vracet přesně stejné
sloupce.
Ale protože máš jedinou tabulku, bude aspoň profiltrovaný obsah pro
všechny ty selecty stejný, takže ho můžeš vyčlenit do společné CTE (to
je ta část za WITH), a v každém z těch selectů ho znovu využít.
V aplikaci pak podle comba a hodnoty dohledáš položku a aktualizuješ u ní číslo. U položek, které ve výsledku obsažené nebudou, nastavíš 0.
Jestli jde tohle nějak udělat LINQem, resp. EF nevím, a i kdyby to šlo, nevím, jak rychlé to bude.
Zobrazeno 7 zpráv z 7.