Dnes ukončíme část seriálu, která se věnuje zacházení se soubory, a
dokončíme ji příkladem toho, jak by se v PHP dal napsat BLOG.
2.8.2004 15:00 | Petr Zajíc | přečteno 56678×
Weblog neboli blog
je, zdá se, fenoménem poslední doby. Zatímco před pár lety chtěl každý
mít svoje osobní stránky, dneska chce každý člověk šmrnclý internetem
mít svůj internetový deníček.
Nejsem tomu osobně nijak nakloněn, ale dá se to použít jako dobrý
příklad na PHP a soubory, takže s chutí do toho.
Pro účely našeho příkladu budeme předpokládat, že každý den
nahrajeme do určité složky na webovém serveru textový soubor s názvem
ve formátu yyyymmdd.txt. Úkolem naší aplikace bude tyto soubory seřadit
a vypsat nějak pěkně do prohlížeče. Aby toho nebylo málo, bude možné
stáhnout si celý obsah blogu jako jeden soubor.
Přenos souborů na server nebudeme řešit. Probírali jsme to v minulém
díle. Nicméně, pro testovací účely jsem připravil několik souborů v mojí složce na serveru,
takže se můžete pokochat (zčásti) pravdivými útržky z mého života v
posledních několika málo dnech. (Připravit
alespoň trochu smysluplné texty byla pro mě nejnáročnější část celého
článku, takže to tak berte. Samozřejmě by soubory mohly obsahovat
cokoli;-))
Vlastní skript bude řešit několik věcí. Zaprvé, projdeme si složku a vypíšeme z ní soubory. Pod každým souborem dole bude ještě zobrazeno datum vypočítané z názvu souboru. Celé to nějak hezky oddělíme. Skript by mohl vypadat zhruba takhle:
<?
$slozka = dir("./blog");
$licha=true;
while($soubor=$slozka->read()) {
if ($soubor=="." || $soubor=="..") continue;
$barva = $licha ? "#FDF5E6" : "#FFFFFF"; // nebo
echo "<div style=\"background-color:$barva\">";
readfile ("./blog/".$soubor);
$datum=explode(".", basename($soubor));
echo "<p
align=\"right\">".date("d.m.Y", strtotime($datum[0]))."</p>";
echo "<hr></div>";
$licha=!$licha;
}
$slozka->close();
?>
Je to na vysvětlení poměrně jednoduché. Skript vezme obsah složky a
načte jej postupně pomocí $slozka->read. Pro lepší čtení budeme u
souborů měnit vždy po jednom dni pozadí, k čemuž nám poslouží proměnná
$licha. Ta se neustále přepíná mezi TRUE a FALSE a podle její hodnoty
nastavujeme pozadí textu souvisejícího s daným dnem.
Vlastní podstatnou část zajišťuje funkce readfile. Hrátky s
basename, explode, strtotime a date slouží jen k tomu, abychom z názvu
souboru dostali česky formátované datum, které pak pod každý den
vypíšeme.
Pozn.: V praxi bychom asi přidali
ještě funkci stránkování, protože by to mohlo být časem dlouhé a
nepřehledné. Taky by bylo potřeba ošetřit řazení souborů. Obojí jsme
vynechali kvůli jednoduchosti. Ke stránkování se ale ještě dostaneme v
některém díle o databázích.
Jako další věc jsme si slíbili, že budeme chtít celý blog stáhnout jako soubor. Blog se bude zvětšovat. To může vést časem k tomu, že by se stahoval extrémně dlouhý soubor, takže přichází na řadu komprese. PHP podporuje kompresi zlib (soubory gz) a prakticky vždy ponechávají administrátoři tuto volbu zapnutou, takže to lze použít. Skript pro kompresi blogu by mohl vypadat následovně:
<?
ob_start();
// tohle je stejné jako v předchozím
případě...
$contents = ob_get_contents();
ob_end_clean();
header("Content-Description: File Transfer");
header("Content-Type: application/force-download");
header("Content-Disposition: attachment; filename=\"blog.gz\"");
echo gzencode($contents);
?>
Ukázat celý skript | Spustit skript
Tady bych měl asi vysvětlit několi věcí. Především, správný
programátor by měl být líný. A když už jsme si předtím náš deníček tak
hezky zformátovali, proč to teď nevyužít? Problém ale spočívá v tom, že
předtím jsme psali pomocí příkazu echo přímo do prohlížeče - jak z toho
tedy teď dostat soubor? Jednoduše - PHP má sadu funkcí umožňujících
bufferování stránky a ukládání do proměnné namísto přímého posílání na
výstup.
Ačkoli to normálně nemusí být vždy ten nejlepší nápad, nám se to
zrovna hodí. Takže, funkce ob_start celý mechanismus začne a zajistí,
že následující příkazy (například echo) se nebudou posílat prohlížeči,
ale ukládat do bufferu. Proběhne totéž jako v prvním příkladu jen s tím
rozdílem, že to, co bylo původně odesláno do prohlížeče je nyní v
proměnné $contents a do prohlížeče se
zatím neposlalo nic! Takže si klidně můžeme dovolit odeslat
hlavičky a následně cokoli. Proč odesíláme právě tyto hlavičky jste se
měli možnost dozvědět v předchozím dílu našeho seriálu.
Celou kompresi pak zajistí funkce gzencode. Ta nejenom provede
kompresi, ale rovněž přidá hlavičky souboru podle specifikace, takže
vznikne skutečný gz soubor. Můžete si jej uložit na disk, rozbalit a
zobrazit. Komprese v našem případě ušetřila asi 50% velikosti souboru,
při větších souborech to ale může být ještě lepší.
Pozn.: PHP může podporovat další
typy kompresí - například bzip nebo zip. V praxi to ale zdaleka nebývá
na serverech povoleno tak často jako gzip.
V několika předchozích dílech jsme hovořili o PHP a souborech.
Krátce shrňme, co podstatného jsme se dozvěděli: