Měla by se binární data ukládat do databáze, nebo ne? A jestliže ano, jak s tím pak pracovat v PHP?
22.9.2004 10:00 | Petr Zajíc | czytane 47670×
RELATED ARTICLES
KOMENTARZE
Tento díl seriálu jste si vyžádali ve svých e-mailových reakcích na
předchozí články. Zaznamenal jsem několik dotazů, jak v PHP a MySQL
pracovat s binárními daty.
Proč to nemám rád
Především, musím se přiznat, že taková řešení moc nepodporuji. Není
mnoho pádných důvodů pro ukládání binárních dat do MySQL databází,
jestliže tvoříte klasickou webovou aplikaci. Budete-li chtít ukládat do
MySQL například sérii obrázků, měli byste vědět, že:
- to je pomalé. Minimálně pomalejší o databázovou režii.
- to zabere víc místa. MySQL musí mít uloženy informace nejen o
samotném souboru, ale i některá metadata.
- to většinou znamená celou řadu práce pro programátora navíc.
Abychom byli upřímní, může to mít i výhody. Přišel jsem na dvě:
- Možná, že binární data v databázi budeme moci lépe zabezpečit,
než kdyby byla na disku.
- Do binárního pole v databázi můžeme uložit prakticky cokoliv. V
jednom řádku dat to může být obrázek, v jiném třeba soubor obsahující
hudbu.
Jestliže ale nemáte jinou možnost, nebo jestliže potřebujete ukládat
binární data, může Vám MySQL sloužit.
Co udělat v databázi
V MySQL je pro uložení binárních dat použit sloupec typu BLOB
(binary large object). Takže, pokud budete chtít definovat tabulku pro
uložení obrázků s jedním sloupcem, můžete použít něco ve smyslu:
CREATE TABLE obrazky
(id INT NOT NULL AUTO_INCREMENT ,
obrazek BLOB NOT NULL , PRIMARY KEY (id));
Jestliže máte definici tabulky hotovou, můžete do ní směle vložit
data. Mějme například ve složce, z níž se skript spouští, uložen soubor
test.jpg. V takovém případě jej můžete načíst pomocí funkce fread do
proměnné a vložit do databáze. Jenže pozor: binární soubor může
obsahovat kdeco, včetně například apostrofů a jiných znaků, které by
při sestavování dotazu mohly databázi zmást. Kdyby se to stalo, uloží
se náš obrázek porušený nebo se dokonce neuloží vůbec. Musíme tedy
použít funkci addslashes, která
vkládaná data před jejich uložením oescapuje. Celé to může vypadat
následovně:
<?
// zde je include souboru s
konstantami
mysql_connect(SQL_HOST, SQL_USERNAME, SQL_PASSWORD);
mysql_select_db(SQL_DBNAME);
$fp = fopen("test.jpg", "rb");
$binarydata = addslashes(fread($fp, filesize("test.jpg")));
mysql_query ("insert into obrazky
(obrazek) values ('" . $binarydata . "')");
fclose($fp);
?>
Pozn.: Někteří tvrdí, že "úpravu"
spočívající v přidání escape znaků pomocí funkce addslashes nepřežijí
obrázky ve formátu jpeg. Sám jsem se s tím nesetkal. Kdyby Vás to
nicméně potkalo, můžete data zakódovat například pomocí funkce base64_encode. Budou ale zabírat v databázi o třetinu
místa více.
Zobrazování binárních dat
Jeden veliký problém spočívá v tom, že musíme vědět, s jakými
binárními daty právě pracujeme, a musíme to sdělit prohlížeči. Takže,
kdybychom měli zobrazovat obrázky uložené v tabulce podle předchozího
příkladu, můžeme směle použít něco jako:
<?
// zde je include souboru s
konstantami
mysql_connect(SQL_HOST, SQL_USERNAME, SQL_PASSWORD);
mysql_select_db(SQL_DBNAME);
if (!isset($_GET["obrazek"])) die ("Nezadali jste
číslo obrázku");
$vysledek = mysql_query ("SELECT * FROM obrazky
where id=".$_GET["obrazek"]);
if (!mysql_num_rows($vysledek)==1) die ("Nemáme takový
obrázek");
header("Content-Type:
image/jpeg");
$row = mysql_fetch_assoc ($vysledek);
echo $row["obrazek"];
?>
Spustit
skript (1) | Spustit
skript (2)
přičemž řádek s voláním funkce header je nezbytný. Jinak totiž
nebohý prohlížeč nepozná, co má s došlými binárními daty dělat.
Co když ale potřebujeme zobrazit nejen samotný obrázek, ale stránku,
která bude obsahovat dejme tomu několik různých obrázků pocházejících z
databáze? To je problém, protože v tom případě nemůžeme použít funkci
header jako v předchozím případě. Máte zhruba tři možnosti:
- Obrázky vyexportovat do souborů a použít je, jako by tam byly
odjakživa (brr, pomalé).
- Znásilnit funkce PHP pro práci s proudy a funkce pro práci s
buffery a nějak to před odesláním binárně poslepovat (brr, složité)
- Použít následující fintu:
<img
src="48_show.php?obrazek=1"><br>
To je moje kočka ;-))<br>
<img src="48_show.php?obrazek=2"><br>
Neuvěřitelná podobnost mého kamaráda a agenta Smithe z
Matrixu.<br>
Spustit
skript
Opravdu to funguje a opravdu v tom není žádná záludnost. Prostě jsme
jako zdroj obrázku nepoužili soubor jpg, ale soubor, který jej "nějak"
generuje.
Závěr
Ukládání binárních dat do databáze je většinou jako technika
kritizováno, ale někdy se to může hodit. Při vkládání dat používejte
addslashes. Pozor, je nutné vědět, jakého typu jsou vkládaná data,
pokud s nimi budete chtít pracovat pomocí PHP a zobrazovat je v
prohlížeči.