<?php
include_once $_SERVER['DOCUMENT_ROOT'] . '/include/shared-manual.inc';
$TOC = array();
$TOC_DEPRECATED = array();
$PARENTS = array();
include_once dirname(__FILE__) ."/toc/book.pdo.inc";
$setup = array (
  'home' => 
  array (
    0 => 'index.php',
    1 => 'PHP Manual',
  ),
  'head' => 
  array (
    0 => 'UTF-8',
    1 => 'de',
  ),
  'this' => 
  array (
    0 => 'pdo.prepared-statements.php',
    1 => 'Prepared Statements und Stored Procedures',
    2 => 'Prepared Statements und Stored Procedures',
  ),
  'up' => 
  array (
    0 => 'book.pdo.php',
    1 => 'PDO',
  ),
  'prev' => 
  array (
    0 => 'pdo.transactions.php',
    1 => 'Transaktionen und auto-commit',
  ),
  'next' => 
  array (
    0 => 'pdo.error-handling.php',
    1 => 'Fehler und Fehlerbehandlung',
  ),
  'alternatives' => 
  array (
  ),
  'source' => 
  array (
    'lang' => 'de',
    'path' => 'reference/pdo/prepared-statements.xml',
  ),
  'history' => 
  array (
  ),
);
$setup["toc"] = $TOC;
$setup["toc_deprecated"] = $TOC_DEPRECATED;
$setup["parents"] = $PARENTS;
manual_setup($setup);

contributors($setup);

?>
<div id="pdo.prepared-statements" class="chapter">
 <h1 class="title">Prepared Statements und Stored Procedures</h1>

 <p class="para">
  Viele der ausgereifteren Datenbanken unterstützen das Konzept der Prepared
  Statements. Was ist das? Man kann sie sich als eine Art von kompiliertem
  Template für das SQL, das eine Anwendung ausführen will, vorstellen, das
  durch variable Parameter angepasst werden kann. Prepared Statements haben
  zwei wichtige Vorteile:
 </p>
 <ul class="itemizedlist">
  <li class="listitem">
   <span class="simpara">
    Die Abfrage muss nur einmal geparst (oder vorbereitet) werden, kann dann
    aber mehrere Male mit denselben oder anderen Parametern ausgeführt werden.
    Wenn die Abfrage vorbereitet wird, kann die Datenbank ihre Vorgehensweise
    zur Ausführung der Abfrage analysieren, kompilieren und optimieren. Für
    komplexe Abfragen kann dieser Vorgang genug Zeit benötigen, dass es eine
    Anwendung merklich verlangsamt, wenn dieselbe Abfrage oft mit
    verschiedenen Parametern wiederholt wird. Mit einem Prepared Statement
    vermeidet die Anwendung den Zyklus der Analyse/Kompilierung/Optimierung.
    Kurz gesagt benötigen Prepared Statements weniger Ressourcen und laufen
    deswegen schneller.
   </span>
  </li>
  <li class="listitem">
   <span class="simpara">
    Die Parameter für Prepared Statements müssen nicht maskiert werden. Der
    Treiber übernimmt das automatisch. Wenn eine Anwendung ausschließlich
    Prepared Statements benutzt, kann sich der Entwickler sicher sein, dass
    keine SQL-Injection auftreten wird. (Wenn aber trotzdem andere Teile der
    Abfrage aus nicht zuverlässigen Eingaben generiert werden, ist dies immer
    noch möglich.)
   </span>
  </li>
 </ul>
 <p class="para">
  Prepared Statements sind so nützlich, dass sie das einzige Feature sind, das
  PDO auch für Treiber emulieren wird, die diese nicht unterstützen. Das
  garantiert, dass eine Anwendung unabhängig von den Möglichkeiten der
  Datenbank dieselbe Art des Datenzugriffs nutzen kann.
 </p>
 <p class="para">
  <div class="example" id="example-1">
   <p><strong>Beispiel #1 Wiederholte Inserts mit Prepared Statements</strong></p>
   <div class="example-contents"><p>
    Dieses Beispiel führt eine INSERT-Abfrage durch, in der ein
    <code class="literal">name</code> und ein <code class="literal">value</code> für die
    benannten Platzhalter eingesetzt werden.
   </p></div>
   <div class="example-contents">
<div class="phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br />$stmt </span><span style="color: #007700">= </span><span style="color: #0000BB">$dbh</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">prepare</span><span style="color: #007700">(</span><span style="color: #DD0000">"INSERT INTO REGISTRY (name, value) VALUES (:name, :value)"</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">$stmt</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">bindParam</span><span style="color: #007700">(</span><span style="color: #DD0000">':name'</span><span style="color: #007700">, </span><span style="color: #0000BB">$name</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">$stmt</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">bindParam</span><span style="color: #007700">(</span><span style="color: #DD0000">':value'</span><span style="color: #007700">, </span><span style="color: #0000BB">$value</span><span style="color: #007700">);<br /><br /></span><span style="color: #FF8000">// eine Zeile einfügen<br /></span><span style="color: #0000BB">$name </span><span style="color: #007700">= </span><span style="color: #DD0000">'one'</span><span style="color: #007700">;<br /></span><span style="color: #0000BB">$value </span><span style="color: #007700">= </span><span style="color: #0000BB">1</span><span style="color: #007700">;<br /></span><span style="color: #0000BB">$stmt</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">execute</span><span style="color: #007700">();<br /><br /></span><span style="color: #FF8000">// eine weitere Zeile mit anderen Werten einfügen<br /></span><span style="color: #0000BB">$name </span><span style="color: #007700">= </span><span style="color: #DD0000">'two'</span><span style="color: #007700">;<br /></span><span style="color: #0000BB">$value </span><span style="color: #007700">= </span><span style="color: #0000BB">2</span><span style="color: #007700">;<br /></span><span style="color: #0000BB">$stmt</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">execute</span><span style="color: #007700">();<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
   </div>

  </div>
 </p>
 <p class="para">
  <div class="example" id="example-2">
   <p><strong>Beispiel #2 Wiederholte Inserts mit Prepared Statements</strong></p>
   <div class="example-contents"><p>
    Dieses Beispiel führt eine INSERT-Abfrage durch, in der ein
    <code class="literal">name</code> und ein <code class="literal">value</code> für die
    positionsabhängigen <code class="literal">?</code>-Platzhalter eingesetzt werden.
   </p></div>
   <div class="example-contents">
<div class="phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br />$stmt </span><span style="color: #007700">= </span><span style="color: #0000BB">$dbh</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">prepare</span><span style="color: #007700">(</span><span style="color: #DD0000">"INSERT INTO REGISTRY (name, value) VALUES (?, ?)"</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">$stmt</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">bindParam</span><span style="color: #007700">(</span><span style="color: #0000BB">1</span><span style="color: #007700">, </span><span style="color: #0000BB">$name</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">$stmt</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">bindParam</span><span style="color: #007700">(</span><span style="color: #0000BB">2</span><span style="color: #007700">, </span><span style="color: #0000BB">$value</span><span style="color: #007700">);<br /><br /></span><span style="color: #FF8000">// eine Zeile einfügen<br /></span><span style="color: #0000BB">$name </span><span style="color: #007700">= </span><span style="color: #DD0000">'one'</span><span style="color: #007700">;<br /></span><span style="color: #0000BB">$value </span><span style="color: #007700">= </span><span style="color: #0000BB">1</span><span style="color: #007700">;<br /></span><span style="color: #0000BB">$stmt</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">execute</span><span style="color: #007700">();<br /><br /></span><span style="color: #FF8000">// eine weitere Zeile mit anderen Werten einfügen<br /></span><span style="color: #0000BB">$name </span><span style="color: #007700">= </span><span style="color: #DD0000">'two'</span><span style="color: #007700">;<br /></span><span style="color: #0000BB">$value </span><span style="color: #007700">= </span><span style="color: #0000BB">2</span><span style="color: #007700">;<br /></span><span style="color: #0000BB">$stmt</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">execute</span><span style="color: #007700">();<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
   </div>

  </div>
 </p>
 <p class="para">
  <div class="example" id="example-3">
   <p><strong>Beispiel #3 Abfragen von Daten mit Prepared Statements</strong></p>
   <div class="example-contents"><p>
    Dieses Beispiel ruft Daten basierend auf einem Schlüsselwert ab, der von
    einem Formular geliefert wird. Die Benutzereingabe wird automatisch
    maskiert, deswegen gibt es kein Risiko eines SQL-Injection-Angriffs.
   </p></div>
   <div class="example-contents">
<div class="phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br />$stmt </span><span style="color: #007700">= </span><span style="color: #0000BB">$dbh</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">prepare</span><span style="color: #007700">(</span><span style="color: #DD0000">"SELECT * FROM REGISTRY where name = ?"</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">$stmt</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">execute</span><span style="color: #007700">([</span><span style="color: #0000BB">$_GET</span><span style="color: #007700">[</span><span style="color: #DD0000">'name'</span><span style="color: #007700">]]);<br />foreach (</span><span style="color: #0000BB">$stmt </span><span style="color: #007700">as </span><span style="color: #0000BB">$row</span><span style="color: #007700">) {<br />  </span><span style="color: #0000BB">print_r</span><span style="color: #007700">(</span><span style="color: #0000BB">$row</span><span style="color: #007700">);<br />}<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
   </div>

  </div>
 </p>
 <p class="para">
  <div class="example" id="example-4">
   <p><strong>Beispiel #4 Eine Stored Procedure mit einem Ausgabeparameter aufrufen</strong></p>
   <div class="example-contents"><p>
    Wenn es der Datenbanktreiber unterstützt, kann eine Anwendung auch
    Parameter für die Ausgabe einführen, ähnlich der Eingabe. Ausgabeparameter
    werden typischerweise benutzt, um Werte von Stored Procedures abzurufen.
    Ausgabeparameter sind etwas komplexer in der Verwendung als
    Eingabeparameter, weil der Entwickler wissen muss, wie groß ein gegebener
    Parameter sein könnte, wenn er ihn einführt. Wenn der Wert sich als
    größer herausstellt als die vorgeschlagene Größe, wird eine Fehlermeldung
    erzeugt.
   </p></div>
   <div class="example-contents">
<div class="phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br />$stmt </span><span style="color: #007700">= </span><span style="color: #0000BB">$dbh</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">prepare</span><span style="color: #007700">(</span><span style="color: #DD0000">"CALL sp_returns_string(?)"</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">$stmt</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">bindParam</span><span style="color: #007700">(</span><span style="color: #0000BB">1</span><span style="color: #007700">, </span><span style="color: #0000BB">$return_value</span><span style="color: #007700">, </span><span style="color: #0000BB">PDO</span><span style="color: #007700">::</span><span style="color: #0000BB">PARAM_STR</span><span style="color: #007700">, </span><span style="color: #0000BB">4000</span><span style="color: #007700">);<br /><br /></span><span style="color: #FF8000">// Aufruf der Stored Procedure<br /></span><span style="color: #0000BB">$stmt</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">execute</span><span style="color: #007700">();<br /><br />print </span><span style="color: #DD0000">"Rückgabewert der Stored Procedure: </span><span style="color: #0000BB">$return_value</span><span style="color: #DD0000">\n"</span><span style="color: #007700">;<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
   </div>

  </div>
 </p>

 <p class="para">
  <div class="example" id="example-5">
   <p><strong>Beispiel #5 Eine Stored Procedure mit einem Eingabe-/Ausgabe-Parameter
    aufrufen</strong></p>
   <div class="example-contents"><p>
    Entwickler können auch Parameter angeben, die Werte für Eingabe und
    Ausgabe enthalten. Die Syntax ist ähnlich den Ausgabeparametern. In diesem
    nächsten Beispiel wird die Zeichenkette &#039;hallo&#039; der Stored Procedure
    übergeben. Wenn diese etwas zurückgibt, wird &#039;hallo&#039; durch den
    Rückgabewert der Stored Procedure ersetzt.
   </p></div>
   <div class="example-contents">
<div class="phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br />$stmt </span><span style="color: #007700">= </span><span style="color: #0000BB">$dbh</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">prepare</span><span style="color: #007700">(</span><span style="color: #DD0000">"CALL sp_takes_string_returns_string(?)"</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">$value </span><span style="color: #007700">= </span><span style="color: #DD0000">'hallo'</span><span style="color: #007700">;<br /></span><span style="color: #0000BB">$stmt</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">bindParam</span><span style="color: #007700">(</span><span style="color: #0000BB">1</span><span style="color: #007700">, </span><span style="color: #0000BB">$value</span><span style="color: #007700">, </span><span style="color: #0000BB">PDO</span><span style="color: #007700">::</span><span style="color: #0000BB">PARAM_STR</span><span style="color: #007700">|</span><span style="color: #0000BB">PDO</span><span style="color: #007700">::</span><span style="color: #0000BB">PARAM_INPUT_OUTPUT</span><span style="color: #007700">, </span><span style="color: #0000BB">4000</span><span style="color: #007700">);<br /><br /></span><span style="color: #FF8000">// Aufruf der Stored Procedure<br /></span><span style="color: #0000BB">$stmt</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">execute</span><span style="color: #007700">();<br /><br />print </span><span style="color: #DD0000">"Rückgabewert der Stored Procedure: </span><span style="color: #0000BB">$value</span><span style="color: #DD0000">\n"</span><span style="color: #007700">;<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
   </div>

  </div>
 </p>
 <p class="para">
  <div class="example" id="example-6">
   <p><strong>Beispiel #6 Ungültige Verwendung von Platzhaltern</strong></p>
   <div class="example-contents">
<div class="phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br />$stmt </span><span style="color: #007700">= </span><span style="color: #0000BB">$dbh</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">prepare</span><span style="color: #007700">(</span><span style="color: #DD0000">"SELECT * FROM REGISTRY where name LIKE '%?%'"</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">$stmt</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">execute</span><span style="color: #007700">([</span><span style="color: #0000BB">$_GET</span><span style="color: #007700">[</span><span style="color: #DD0000">'name'</span><span style="color: #007700">]]);<br /><br /></span><span style="color: #FF8000">// Platzhalter müssen anstelle des ganzen Wertes verwendet werden<br /></span><span style="color: #0000BB">$stmt </span><span style="color: #007700">= </span><span style="color: #0000BB">$dbh</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">prepare</span><span style="color: #007700">(</span><span style="color: #DD0000">"SELECT * FROM REGISTRY where name LIKE ?"</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">$stmt</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">execute</span><span style="color: #007700">([</span><span style="color: #DD0000">"%</span><span style="color: #0000BB">$_GET</span><span style="color: #007700">[</span><span style="color: #0000BB">name</span><span style="color: #007700">]</span><span style="color: #DD0000">%"</span><span style="color: #007700">]);<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
   </div>

  </div>
 </p>
</div>
<?php manual_footer($setup); ?>