PHP (48) - Práce s binárními daty (BLOB)

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 | přečteno 47434×

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:

Abychom byli upřímní, může to mít i výhody. Přišel jsem na dvě:

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:

<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.

Online verze článku: http://www.linuxsoft.cz/article.php?id_article=420