Článek vyšel v rámci PHP okénka na serveru Root.cz.
Řekněme, že chceme uživateli na stránce elektronického obchodu zobrazit posledních pět výrobků, které si prohlížel. Pro uložení takovéto informace se ideálně hodí session proměnná:
<?php // $id je ID právě prohlíženého záznamu if (!isset($_SESSION["prohlednute"])) { $_SESSION["prohlednute"] = array(); // prvotní inicializace pole } elseif (($pos = array_search($id, $_SESSION["prohlednute"])) !== false) { // $id už v poli je unset($_SESSION["prohlednute"][$pos]); // vyjmeme ho a později vložíme na konec } else { unset($_SESSION["prohlednute"][4]); // zobrazovat pouze posledních 5 prohlédnutých výrobků } // na tomto místě bude výpis prohlédnutých záznamů array_unshift($_SESSION["prohlednute"], $id); // uložíme aktuální zboží pro příští stránku ?>
Výpis prohlednutých záznamů by byl triviální za předpokladu, že bychom zboží nechtěli řadit podle pořadí, ve kterém bylo prohlédnuté. Pokud ho tak ale řadit chceme, budeme se muset trochu snažit:
<?php if ($_SESSION["prohlednute"]) { $vyrobky = array(); $result = mysql_query("SELECT * FROM vyrobky WHERE id IN (" . implode(", ", $_SESSION["prohlednute"]) . ")"); while ($row = mysql_fetch_assoc($result)) { $vyrobky[$row["id"]] = $row; } mysql_free_result($result); foreach ($_SESSION["prohlednute"] as $id) { // zobrazení záznamu $vyrobky[$id] } } ?>
Kód je ale neelegantní a náročný na paměť. Problém se dá mnohem elegantněji vyřešit s využitím MySQL funkce FIELD. Tato funkce přijímá seznam parametrů a vrací pořadí prvního z nich mezi ostatními. Např. FIELD(id, 2, 4, 6)
tedy vrátí pro id=4
dvojku, protože 4
je mezi dalšími parametry na druhém místě. A to je přesně ta funkce, kterou potřebujeme – výsledek chceme setřídit podle pořadí id
v poli $_SESSION["prohlednute"]
.
<?php if ($_SESSION["prohlednute"]) { $prohlednute_list = implode(", ", $_SESSION["prohlednute"]); $result = mysql_query("SELECT * FROM vyrobky WHERE id IN ($prohlednute_list) ORDER BY FIELD(id, $prohlednute_list)"); while ($row = mysql_fetch_assoc($result)) { // zobrazení záznamu $row } mysql_free_result($result); } ?>
Třídění podle dynamického výrazu je ještě pomalejší než třídění podle neindexovaného sloupce, takže bychom se mu za normálních okolností měli vyhnout. Na tomto místě si ho ale můžeme dovolit, protože počet vrácených záznamů je pevně omezen.
Přijďte si o tomto tématu popovídat na školení Návrh a používání MySQL databáze.
Diskuse je zrušena z důvodu spamu.