Php 5 bringt eine komplette Reflection API mit, die Möglichkeiten bietet, um
Klassen, Interfaces, Funktionen und Methoden ebenso wie auch Extensions
zurückzuentwickeln. Zusätzlich bietet die Reflection API Wege,
dokumentarische Kommentare von Funktionen, Klassen und Methoden abzufragen.
Die Refelection API ist eine objektorientierte Erweiterung der Zend Engine,
bestehend aus den folgenden Klassen:
<?php class Reflection { } interface Reflector { } class ReflectionException extends Exception { } class ReflectionFunction implements Reflector { } class ReflectionParameter implements Reflector { } class ReflectionMethod extends ReflectionFunction { } class ReflectionClass implements Reflector { } class ReflectionObject extends ReflectionClass { } class ReflectionProperty implements Reflector { } class ReflectionExtension implements Reflector { } ?>
Anmerkung:
Für detailierte Erläuterungen dieser Klassen werfen Sie einen Blick auf die
nächsten Kapitel.
Wenn wir den unten stehenden Beispielcode ausführen würden:
Beispiel 19-33. Grundlegende Nutzung der Reflection API
ReflectionException erweitert die normale
Exception und wird von der
Reflection API geworfen. Es werden keine spezifischen Methoden oder
Eigenschaften eingeführt.
Die ReflectionFunction Klasse lässt Sie
Funktionen zurückentwickeln.
<?php class ReflectionFunction implements Reflector { final private __clone() public object __construct(string name) public string __toString() public static string export(string name, bool return) public string getName() public bool isInternal() public bool isUserDefined() public string getFileName() public int getStartLine() public int getEndLine() public string getDocComment() public array getStaticVariables() public mixed invoke(mixed args) public mixed invokeArgs(array args) public bool returnsReference() public ReflectionParameter[] getParameters() public int getNumberOfParameters() public int getNumberOfRequiredParameters() } ?>
Anmerkung: getNumberOfParameters() und
getNumberOfRequiredParameters() wurden in PHP
5.0.3 hinzugefügt, während invokeArgs() in PHP 5.1.0
neu hinzukam.
Um einen Blick in eine Funktion zu werfen, muss man erst eine Instanz der Klasse
ReflectionFunction erzeugen. Danach kann man jede
der oben aufgeführten Methoden dieser Instanz aufrufen.
Beispiel 19-34. Die ReflectionFunction Klasse benutzen
<?php /** * Ein einfacher Zähler * * @return int */ function counter() { static $c = 0; return $c++; }
// Erzeuge eine neue Instanz der ReflectionFunction Klasse $func = new ReflectionFunction('counter');
// Grundlegende Informationen ausgeben printf( "===> Die %s Funktion '%s'\n". " ist deklariert in %s\n". " Zeilen %d bis %d\n", $func->isInternal() ? 'interne' : 'benutzerdefinierte', $func->getName(), $func->getFileName(), $func->getStartLine(), $func->getEndline() );
// Statische Variablen ausgeben, wenn welche existieren if ($statics = $func->getStaticVariables()) { printf("---> Statische Variablen: %s\n", var_export($statics, 1)); }
// Die Funktion aufrufen printf("---> Der Aufruf gibt zurück: "); var_dump($func->invoke());
// Sie könnten bevorzugen die export() Methode zu verwenden echo "\nReflectionFunction::export() gibt zurück:\n"; echo ReflectionFunction::export('counter'); ?>
Anmerkung:
Die Methode invoke() akzeptiert eine variable Anzahl
von Argumenten, welche genau wie von call_user_func()
an die Funktion übergeben werden.
Die ReflectionParameter Klasse frägt Informationen
über die Parameter einer Funktion oder Methode ab.
<?php class ReflectionParameter implements Reflector { final private __clone() public object __construct(string name) public string __toString() public static string export(mixed function, mixed parameter, bool return) public string getName() public bool isPassedByReference() public ReflectionFunction getDeclaringFunction() public ReflectionClass getDeclaringClass() public ReflectionClass getClass() public bool isArray() public bool allowsNull() public bool isPassedByReference() public bool getPosition() public bool isOptional() public bool isDefaultValueAvailable() public mixed getDefaultValue() } ?>
Anmerkung: getDefaultValue(),
isDefaultValueAvailable() und
isOptional() wurden in PHP 5.0.3 hinzugefügt,
während isArray() in PHP 5.1.0 neu ist.
getDeclaringFunction() und
getPosition() wurden in PHP 5.1.3 hinzugefügt.
Um einen Blick in Funktionsparameter zu werfen, muss man zuerst eine Instanz
der Klassen ReflectionFunction oder
ReflectionMethod erzeugen und kann dann deren
getParameters() Methode verwenden, um ein Array ihrer
Parameter abzufragen.
Beispiel 19-35. Die ReflectionParameter Klasse verwenden
<?php function foo($a, $b, $c) { } function bar(Exception $a, &$b, $c) { } function baz(ReflectionFunction $a, $b = 1, $c = null) { } function abc() { }
// Erzeuge eine Instanz von ReflectionFunction mit dem // auf der Kommandozeile übergebenen Parameter $reflect = new ReflectionFunction($argv[1]);
Die ReflectionClass Klasse erlaubt es, Klassen
zurückzuentwickeln.
<?php class ReflectionClass implements Reflector { final private __clone() public object __construct(string name) public string __toString() public static string export(mixed class, bool return) public string getName() public bool isInternal() public bool isUserDefined() public bool isInstantiable() public bool hasConstant(string name) public bool hasMethod(string name) public bool hasProperty(string name) public string getFileName() public int getStartLine() public int getEndLine() public string getDocComment() public ReflectionMethod getConstructor() public ReflectionMethod getMethod(string name) public ReflectionMethod[] getMethods() public ReflectionProperty getProperty(string name) public ReflectionProperty[] getProperties() public array getConstants() public mixed getConstant(string name) public ReflectionClass[] getInterfaces() public bool isInterface() public bool isAbstract() public bool isFinal() public int getModifiers() public bool isInstance(stdclass object) public stdclass newInstance(mixed args) public stdclass newInstanceArgs(array args) public ReflectionClass getParentClass() public bool isSubclassOf(ReflectionClass class) public array getStaticProperties() public mixed getStaticPropertyValue(string name [, mixed default]) public void setStaticPropertyValue(string name, mixed value) public array getDefaultProperties() public bool isIterateable() public bool implementsInterface(string name) public ReflectionExtension getExtension() public string getExtensionName() } ?>
Anmerkung: hasConstant(), hasMethod(),
hasProperty(), getStaticPropertyValue()
und setStaticPropertyValue() wurden in PHP 5.1.0
hinzugefügt, während newInstanceArgs() in PHP 5.1.3 dazu
kam.
Um einen Blick in eine Klasse zu werfen, muss man zuerst eine Instanz der
Klasse ReflectionClass erzeugen. Man kann danach jede
der oben angeführten Methoden der Instanz verwenden.
Beispiel 19-36. Die ReflectionClass Klasse verwenden
echo "\n---> ist new Object() Instanz? "; echo $class->isInstance(new Object()) ? 'ja' : 'nein'; } ?>
Anmerkung:
Die Methode newInstance() akzeptiert eine variable
Anzahl von Argumenten, welche der Funktion genau wie in
call_user_func() übergeben werden.
Anmerkung: $class = new ReflectionClass('Foo'); $class->isInstance($arg)
ist äquivalent zu $arg instanceof Foo oder
is_a($arg, 'Foo').
Die Klasse ReflectionObject erlaubt das Zurückentwickeln
von Objekten.
<?php class ReflectionObject extends ReflectionClass { final private __clone() public object __construct(mixed object) public string __toString() public static string export(mixed object, bool return) } ?>
Die ReflectionMethod Klasse erlaubt es,
Klassenmethoden zurückzuentwickeln.
<?php class ReflectionMethod extends ReflectionFunction { public __construct(mixed class, string name) public string __toString() public static string export(mixed class, string name, bool return) public mixed invoke(stdclass object, mixed args) public mixed invokeArgs(stdclass object, array args) public bool isFinal() public bool isAbstract() public bool isPublic() public bool isPrivate() public bool isProtected() public bool isStatic() public bool isConstructor() public bool isDestructor() public int getModifiers() public ReflectionClass getDeclaringClass()
// Ererbt von ReflectionFunction final private __clone() public string getName() public bool isInternal() public bool isUserDefined() public string getFileName() public int getStartLine() public int getEndLine() public string getDocComment() public array getStaticVariables() public bool returnsReference() public ReflectionParameter[] getParameters() public int getNumberOfParameters() public int getNumberOfRequiredParameters() } ?>
Um einen Blick in eine Methode zu werfen, muss man zuerst eine Instanz der
ReflectionMethod Klasse erzeugen. Man kann dann
jede der oben aufgeführten Methoden dieser Instanz aufrufen.
Beispiel 19-37. Die ReflectionMethod Klasse verwenden
<?php class Counter { private static $c = 0;
/** * Zähler erhöhen * * @final * @static * @access public * @return int */ final public static function increment() { return ++self::$c; } }
// Erzeuge eine Instanz der ReflectionMethod Klasse $method = new ReflectionMethod('Counter', 'increment');
// Statische Variablen ausgeben, falls welche existieren if ($statics= $method->getStaticVariables()) { printf("---> Statische Variablen: %s\n", var_export($statics, 1)); }
// Die Methode aufrufen printf("---> Aufruf gibt zurück: "); var_dump($method->invoke(NULL)); ?>
Anmerkung:
Der Versuch, eine private, protected oder abstract Methode aufzurufen,
wird darin enden, dass eine Exception aus der invoke()
Methode geworfen wird.
Anmerkung:
Für statische Methoden sollten Sie, wie man oben sieht, NULL als erstes
Argument von invoke() übergeben. Für nicht statische
Methoden übergeben Sie eine Instanz der Klasse.
Die ReflectionProperty Klasse lässt Sie
Klasseneigenschaften zurückentwickeln.
<?php class ReflectionProperty implements Reflector { final private __clone() public __construct(mixed class, string name) public string __toString() public static string export(mixed class, string name, bool return) public string getName() public bool isPublic() public bool isPrivate() public bool isProtected() public bool isStatic() public bool isDefault() public int getModifiers() public mixed getValue(stdclass object) public void setValue(stdclass object, mixed value) public ReflectionClass getDeclaringClass() public string getDocComment() } ?>
Anmerkung: getDocComment() wurde in PHP 5.1.0 hinzugefügt.
Um einen Blick in eine Eigenschaft zu werfen, müssen Sie zuerst eine
Instanz der Klasse ReflectionProperty erzeugen.
Danach können Sie jede der oben angeführten Methoden der Instanz aufrufen.
Beispiel 19-38. Die ReflectionProperty Klasse verwenden
<?php class String { public $laenge = 5; }
// Erzeuge eine Instanz der Klasse ReflectionPropety $prop = new ReflectionProperty('String', 'length');
Anmerkung:
Der Versuch, den Wert einer private oder protected Klasseneigenschaft
zu lesen oder zu setzen, wird darin enden, dass eine Exception geworfen
wird.
Die ReflectionExtension Klasse lässt Sie
Extensions zurückentwickeln. Man kann alle geladenen Extensions zur
Laufzeit mittels get_loaded_extensions() abrufen.
<?php class ReflectionExtension implements Reflector { final private __clone() public __construct(string name) public string __toString() public static string export(string name, bool return) public string getName() public string getVersion() public ReflectionFunction[] getFunctions() public array getConstants() public array getINIEntries() public ReflectionClass[] getClasses() public array getClassNames() } ?>
Um einen Blick in eine Extension zu werfen, muss man zuerst eine Instanz
der Klasse ReflectionExtension erzeugen. Danach kann
man alle oben aufgeführten Methoden der Instanz aufrufen.
Beispiel 19-39. Die ReflectionExtension Klasse verwenden
<?php // Erzeuge eine neue Instanz der ReflectionExtension Klasse $ext = new ReflectionExtension('standard');
Für den Fall, dass Sie spezialisierte Versionen der eingebauten Klassen
erzeugen wollen (zum Beispiel, um kolorierte HTML Ausgaben zu erzeugen oder
leicht zugängliche Klassenvariablen oder -methoden oder Hilfmethoden
hinzuzufügen), können Sie fortfahren und die Klassen ableiten.
Beispiel 19-40. Erweiterung der eingebauten Klassen
<?php /** * M_Reflection_Method Klasse */ class My_Reflection_Method extends ReflectionMethod { public $visibility = '';
public function __construct($o, $m) { parent::__construct($o, $m); $this->visibility= Reflection::getModifierNames($this->getModifiers()); } }
/** * Demoklasse #1 * */ class T { protected function x() {} }
/** * Demoklasse #2 * */ class U extends T { function x() {} }
Anmerkung:
Achtung: Wenn Sie den Konstruktor überschreiben, denken Sie daran,
den Vaterkonstruktor _vor_ jedem anderen Code den Sie hinzufügen aufzurufen.
Unterlassung wird zu folgendem Ergebnis führen:
Fatal error: Internal error: Failed to retrieve the reflection object
Dieses Tutorial beschreibt sehr gut die Wirkunsweise von objektorientiertes Programmieren. Also bestens geeignet um das objektorientierte Programmieren zu verstehen.
In mehreren Tutorials wird der Zugriff auf Facebook Daten mittels Graph API, FQL und REST API erklärt. Alle Codebeispiele liegen zum Ausprobieren in einem SVN, bzw. github Repository bereit.