-
Notifications
You must be signed in to change notification settings - Fork 43
/
Copy pathexceptions.xml
583 lines (524 loc) · 16.6 KB
/
exceptions.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
<?xml version="1.0" encoding="utf-8"?>
<!-- EN-Revision: 844190217fcb8ad1d0790ef5a90f7c18fa9d1a9d Maintainer: sammywg Status: ready -->
<!-- Reviewed: yes -->
<!-- Rev-Revision: 2e8ef0a1bd98243cb2c6c5c627a195bb53a7a440 Reviewer: samesch -->
<chapter xml:id="language.exceptions" xmlns="http://docbook.org/ns/docbook">
<title>Exceptions (Ausnahmen)</title>
<para>
PHP hat ein Exceptionmodell ähnlich dem anderer Programmiersprachen. Eine
Exception kann in PHP ausgelöst (&throw;) und abgefangen (&catch;) werden.
Um das Abfangen potentieller Exceptions zu ermöglichen, sollte der jeweilige
Code von einem &try;-Block umschlossen werden. Jeder &try;-Block muss
mindestens einen zugehörigen &catch;- oder &finally;-Block besitzen.
</para>
<para>
Wenn eine Exception ausgelöst wird und der aktuelle Funktionsbereich keinen
&catch;-Block hat, steigt die Exception im Aufrufstapel bis zur aufrufenden
Funktion auf, bis sie einen passenden &catch;-Block findet. Alle
&finally;-Blöcke, auf die sie unterwegs trifft, werden ausgeführt. Wenn der
Aufrufstapel bis in den globalen Bereich abgewickelt ist, ohne auf einen
passenden &catch;-Block zu stoßen, bricht das Programm mit einem fatalen
Fehler ab, es sei denn, es wurde ein globaler Exception-Handler gesetzt.
</para>
<para>
Das ausgelöste Objekt muss eine Instanz von (&instanceof;)
<interfacename>Throwable</interfacename> sein. Der Versuch ein Objekt
auszulösen, das das nicht ist, wird einen fatalen PHP-Fehler zur Folge
haben.
</para>
<para>
Seit PHP 8.0.0 ist das Schlüsselwort &throw; ein Ausdruck und kann in jedem
Ausdruckskontext verwendet werden. In früheren Versionen war es eine
Anweisung und musste in einer eigenen Zeile stehen.
</para>
<sect1 annotations="chunk:false" xml:id="language.exceptions.catch">
<title><literal>catch</literal></title>
<para>
Ein &catch;-Block definiert, wie auf eine ausgelöste Exception reagiert
werden soll. Ein &catch;-Block definiert eine oder mehrere Arten von
Exceptions oder Fehlern, die er behandeln kann, und optional eine Variable,
der die Exception zugewiesen werden soll (vor PHP 8.0.0 war die Variable
erforderlich). Der erste &catch;-Block, auf den eine ausgelöste Exception
oder ein Fehler trifft, der mit dem Typ des ausgelösten Objekts
übereinstimmt, behandelt das Objekt.
</para>
<para>
Mehrere &catch;-Blöcke können verwendet werden, um verschiedene Klassen von
Exceptions abzufangen. Wenn innerhalb des &try;-Blocks keine Exception
ausgelöst wird, wird die normale Programmausführung nach dem letzten in
Folge definierten &catch;-Block fortgesetzt. Exceptions können innerhalb
eines &catch;-Blocks ausgelöst (oder erneut ausgelöst) werden. Falls nicht,
wird die Ausführung nach dem &catch;-Block, der ausgelöst wurde,
fortgesetzt.
</para>
<para>
Wenn eine Exception ausgelöst wird, führt PHP den Programmcode hinter der
auslösenden Anweisung nicht aus, sondern versucht, den ersten passenden
&catch;-Block zu finden. Falls eine Exception nicht abgefangen wird, wird
ein fataler Fehler mit einer
"<literal>Uncaught Exception ...</literal>"-Nachricht ausgegeben, sofern
keine Behandlung mittels <function>set_exception_handler</function>
definiert wurde.
</para>
<para>
Seit PHP 7.1.0 kann ein &catch;-Block mehrere Exceptions getrennt durch
Pipe-Zeichen (<literal>|</literal>) angeben. Dies ist nützlich, wenn
unterschiedliche Exceptions von unterschiedlichen Klassenhierarchien gleich
behandelt werden sollen.
</para>
<para>
Seit PHP 8.0.0 ist der Variablenname für eine abgefangene Exception
optional. Wird er nicht angegeben, wird der &catch;-Block trotzdem
ausgeführt, hat aber keinen Zugriff auf das ausgelöste Objekt.
</para>
</sect1>
<sect1 annotations="chunk:false" xml:id="language.exceptions.finally">
<title><literal>finally</literal></title>
<para>
Ein &finally;-Block kann auch nach den &catch;-Blöcken oder stattdessen
definiert werden. Egal, ob eine Exception ausgelöst wurde, wird der Code
innerhalb des &finally;-Blocks immer nach den &try;- und &catch;-Blöcken
ausgeführt, bevor die normale Ausführung fortgesetzt wird.
</para>
<para>
Eine erwähnenswerte Wechselwirkung besteht zwischen dem &finally;-Block
und einer &return;-Anweisung. Wird eine &return;-Anweisung innerhalb der
&try;- oder &catch;-Blöcke angetroffen, wird der &finally;-Block dennoch
ausgeführt. Außerdem wird die &return;-Anweisung ausgewertet, wenn sie
angetroffen wird, aber das Ergebnis wird erst nach dem &finally;-Block
zurückgegeben. Des Weiteren wird, wenn der &finally;-Block ebenfalls eine
&return;-Anweisung enthält, der Wert aus dem &finally;-Block zurückgegeben.
</para>
</sect1>
<sect1 annotations="chunk:false" xml:id="language.exceptions.exception-handler">
<title>Der globale Exception-Handler</title>
<para>
Wenn eine Exception in den globalen Bereich aufsteigen darf, kann sie durch
einen globalen Exception-Handler abgefangen werden, falls gesetzt. Die
Funktion <function>set_exception_handler</function> kann eine Funktion
festlegen, die anstelle eines &catch;-Blocks aufgerufen wird, wenn kein
anderer Block aufgerufen wird. Der Effekt ist im Wesentlichen derselbe, als
ob das gesamte Programm in einen &try;-&catch;-Block mit dieser Funktion
als &catch; verpackt wäre.
</para>
</sect1>
<sect1 annotations="chunk:false" xml:id="language.exceptions.notes">
&reftitle.notes;
<note>
<para>
Interne PHP-Funktionen verwenden in den meisten Fällen
<link linkend="ini.error-reporting">Error-Reporting</link>, nur moderne
<link linkend="language.oop5">objektorientierte</link> Erweiterungen
nutzen Exceptions. Fehler können allerdings einfach mittels
<link linkend="class.errorexception">ErrorException</link> in eine
Exception umgewandelt werden. Diese Technik funktioniert jedoch nur bei
nicht-fatalen Fehlern.
</para>
<example>
<title>Fehlermeldungen in Exceptions umwandeln</title>
<programlisting role="php">
<![CDATA[
<?php
function exceptions_error_handler($severity, $message, $filename, $lineno) {
throw new ErrorException($message, 0, $severity, $filename, $lineno);
}
set_error_handler('exceptions_error_handler');
?>
]]>
</programlisting>
</example>
</note>
<tip>
<para>
Die <link linkend="intro.spl">Standard PHP Library (SPL)</link> bietet
eine große Anzahl
<link linkend="spl.exceptions">eingebauter Exceptions</link>.
</para>
</tip>
</sect1>
<sect1 annotations="chunk:false" xml:id="language.exceptions.examples">
&reftitle.examples;
<example>
<title>Eine Exception auslösen</title>
<programlisting role="php">
<![CDATA[
<?php
function inverse($x) {
if (!$x) {
throw new Exception('Division durch Null.');
}
return 1/$x;
}
try {
echo inverse(5) . "\n";
echo inverse(0) . "\n";
} catch (Exception $e) {
echo 'Exception abgefangen: ', $e->getMessage(), "\n";
}
// Ausführung fortsetzen
echo "Hallo Welt\n";
?>
]]>
</programlisting>
&example.outputs;
<screen>
<![CDATA[
0.2
Exception abgefangen: Division durch Null
Hallo Welt
]]>
</screen>
</example>
<example>
<title>Exceptionbehandlung mit einem &finally;-Block</title>
<programlisting role="php">
<![CDATA[
<?php
function inverse($x) {
if (!$x) {
throw new Exception('Division durch Null.');
}
return 1/$x;
}
try {
echo inverse(5) . "\n";
} catch (Exception $e) {
echo 'Exception abgefangen: ', $e->getMessage(), "\n";
} finally {
echo "Erstes finally.\n";
}
try {
echo inverse(0) . "\n";
} catch (Exception $e) {
echo 'Exception abgefangen: ', $e->getMessage(), "\n";
} finally {
echo "Zweites finally.\n";
}
// Ausführung fortsetzen
echo "Hallo Welt\n";
?>
]]>
</programlisting>
&example.outputs;
<screen>
<![CDATA[
0.2
Erstes finally.
Exception abgefangen: Division durch Null.
Zweites finally.
Hallo Welt
]]>
</screen>
</example>
<example>
<title>Wechselwirkung zwischen dem &finally;-Block und &return;</title>
<programlisting role="php">
<![CDATA[
<?php
function test() {
try {
throw new Exception('foo');
} catch (Exception $e) {
return 'catch';
} finally {
return 'finally';
}
}
echo test();
?>
]]>
</programlisting>
&example.outputs;
<screen>
<![CDATA[
finally
]]>
</screen>
</example>
<example>
<title>Verschachtelte Exceptions</title>
<programlisting role="php">
<![CDATA[
<?php
class MyException extends Exception { }
class Test {
public function testing() {
try {
try {
throw new MyException('foo!');
} catch (MyException $e) {
// Exception erneut auslösen
throw $e;
}
} catch (Exception $e) {
var_dump($e->getMessage());
}
}
}
$foo = new Test;
$foo->testing();
?>
]]>
</programlisting>
&example.outputs;
<screen>
<![CDATA[
string(4) "foo!"
]]>
</screen>
</example>
<example>
<title>Behandlung mehrerer Exceptions in einem Catch-Block</title>
<programlisting role="php">
<![CDATA[
<?php
class MyException extends Exception { }
class MyOtherException extends Exception { }
class Test {
public function testing() {
try {
throw new MyException();
} catch (MyException | MyOtherException $e) {
var_dump(get_class($e));
}
}
}
$foo = new Test;
$foo->testing();
?>
]]>
</programlisting>
&example.outputs;
<screen>
<![CDATA[
string(11) "MyException"
]]>
</screen>
</example>
<example>
<title>Catch-Block ohne Angabe einer Variablen</title>
<para>Erst ab PHP 8.0.0 erlaubt.</para>
<programlisting role="php">
<![CDATA[
<?php
class SpecificException extends Exception {}
function test() {
throw new SpecificException('Oopsie');
}
try {
test();
} catch (SpecificException) {
print "Eine SpecificException wurde ausgelöst, aber die Details interessieren uns nicht.";
}
?>
]]>
</programlisting>
&example.outputs;
<screen>
<![CDATA[
Eine SpecificException wurde ausgelöst, aber die Details interessieren uns nicht.
]]>
</screen>
</example>
<example>
<title>Als Ausdruck Auslösen</title>
<para>Erst ab PHP 8.0.0 erlaubt.</para>
<programlisting role="php">
<![CDATA[
<?php
function test() {
do_something_risky() or throw new Exception('Es hat nicht funktioniert');
}
function do_something_risky() {
return false; // Fehler simulieren
}
try {
test();
} catch (Exception $e) {
print $e->getMessage();
}
?>
]]>
</programlisting>
&example.outputs;
<screen>
<![CDATA[
Es hat nicht funktioniert
]]>
</screen>
</example>
</sect1>
<sect1 xml:id="language.exceptions.extending">
<title>Exceptions erweitern</title>
<para>
Eine benutzerdefinierte Exceptionklasse kann durch Ableitung von der
eingebauten Exceptionklasse erstellt werden. Die unten angegebenen Methoden
und Eigenschaften zeigen, was innerhalb der Kindklasse von der eingebauten
Exceptionklasse verfügbar ist.
</para>
<example>
<title>Die eingebaute Exceptionklasse</title>
<programlisting role="php">
<![CDATA[
<?php
class Exception implements Throwable
{
protected $message = 'Unknown exception'; // Exceptionmitteilung
private $string; // __toString-Cache
protected $code = 0; // Benutzerdefinierte Fehlernummer
protected $file; // Quelldateiname der Exception
protected $line; // Quelldateizeile der Exception
private $trace; // Rückverfolgung
private $previous; // Vorherige Exception, falls verschachtelte Exception
public function __construct($message = '', $code = 0, Throwable $previous = null);
final private function __clone(); // Verhindert klonen von Exceptions
final public function getMessage(); // Mitteilung der Exception
final public function getCode(); // Fehlercode der Exception
final public function getFile(); // Quelldateiname
final public function getLine(); // Quelldateizeile
final public function getTrace(); // Array zum Rückverfolgen
final public function getPrevious(); // Vorherige Exception
final public function getTraceAsString(); // Formatierter String der Rückverfolgung
// Überschreibbar
public function __toString(); // Formatierter String für die Ausgabe
}
?>
]]>
</programlisting>
</example>
<para>
Wenn eine Klasse die eingebaute Exceptionklasse erweitert und den
<link linkend="language.oop5.decon">Konstruktor</link> neu definiert, ist
es dringend empfohlen, dass der Konstruktor der Klasse
<link linkend="language.oop5.paamayim-nekudotayim">parent::__construct()</link>
aufruft, um sicherzustellen, dass alle verfügbaren Daten korrekt zugewiesen
wurden. Die
<link linkend="language.oop5.magic">__toString()</link>-Methode kann
überschrieben werden, um eine benutzerdefinierte Ausgabe anzubieten, wenn
das Objekt durch eine Zeichenkette repräsentiert werden soll.
</para>
<note>
<para>
Exceptions können nicht geklont werden. Der Versuch, eine Exception zu
<link linkend="language.oop5.cloning">klonen</link>, wird einen fatalen
<constant>E_ERROR</constant>-Fehler zur Folge haben.
</para>
</note>
<example>
<title>Die Exceptionklasse erweitern</title>
<programlisting role="php">
<![CDATA[
<?php
/**
* Eine eigene Exceptionklasse definieren
*/
class MyException extends Exception
{
// Die Exception neu definieren, damit die Mitteilung nicht optional ist
public function __construct($message, $code = 0, Throwable $previous = null) {
// etwas Code
// sicherstellen, dass alles korrekt zugewiesen wird
parent::__construct($message, $code, $previous);
}
// benutzerdefinierte Stringdarstellung des Objektes
public function __toString() {
return __CLASS__ . ": [{$this->code}]: {$this->message}\n";
}
public function customFunction() {
echo "Eine eigene Funktion für diesen Exceptiontyp\n";
}
}
/**
* Erzeuge eine Klasse, um die Exception zu testen
*/
class TestException
{
public $var;
const THROW_NONE = 0;
const THROW_CUSTOM = 1;
const THROW_DEFAULT = 2;
function __construct($avalue = self::THROW_NONE) {
switch ($avalue) {
case self::THROW_CUSTOM:
// eigene Exception auslösen
throw new MyException('1 ist ein ungültiger Parameter', 5);
break;
case self::THROW_DEFAULT:
// Vorgabe werfen
throw new Exception('2 ist kein zugelassener Parameter', 6);
break;
default:
// Keine Exception; das Objekt wird erzeugt
$this->var = $avalue;
break;
}
}
}
// Beispiel 1
try {
$o = new TestException(TestException::THROW_CUSTOM);
} catch (MyException $e) { // Wird gefangen
echo "Meine Exception gefangen\n", $e;
$e->customFunction();
} catch (Exception $e) { // Übersprungen
echo "Standardexception gefangen\n", $e;
}
// Ausführung fortsetzen
var_dump($o); // Null
echo "\n\n";
// Beispiel 2
try {
$o = new TestException(TestException::THROW_DEFAULT);
} catch (MyException $e) { // Dieser Typ passt nicht
echo "Meine Exception gefangen\n", $e;
$e->customFunction();
} catch (Exception $e) { // Wird gefangen
echo "Standardexception gefangen\n", $e;
}
// Ausführung fortsetzen
var_dump($o); // Null
echo "\n\n";
// Beispiel 3
try {
$o = new TestException(TestException::THROW_CUSTOM);
} catch (Exception $e) { // Wird gefangen
echo "Standardexception gefangen\n", $e;
}
// Ausführung fortsetzen
var_dump($o); // Null
echo "\n\n";
// Beispiel 4
try {
$o = new TestException();
} catch (Exception $e) { // Übersprungen, keine Exception ausgelöst
echo "Standardexception gefangen\n", $e;
}
// Ausführung fortsetzen
var_dump($o); // TestException
echo "\n\n";
?>
]]>
</programlisting>
</example>
</sect1>
</chapter>
<!-- Keep this comment at the end of the file
Local variables:
mode: sgml
sgml-omittag:t
sgml-shorttag:t
sgml-minimize-attributes:nil
sgml-always-quote-attributes:t
sgml-indent-step:1
sgml-indent-data:t
indent-tabs-mode:nil
sgml-parent-document:nil
sgml-default-dtd-file:"~/.phpdoc/manual.ced"
sgml-exposed-tags:nil
sgml-local-catalogs:nil
sgml-local-ecat-files:nil
End:
vim600: syn=xml fen fdm=syntax fdl=2 si
vim: et tw=78 syn=sgml
vi: ts=1 sw=1
-->