7.12 Dynamisches Binden 

Mit allzu vielen Neuerungen wartet C# 4.0 nicht auf. Man hat den Eindruck, dass diese Sprache »rund« ist, den allgemeinen Anforderungen genügt. Das war auch bereits beim Erscheinen von C# 3.0 ähnlich, obwohl in der Vorgängerversion mit LINQ eine neue Technologie eingeführt worden ist, die manche Spracherweiterung mit sich brachte, die eine Ergänzung der .NET-Sprachen erforderlich machte. Im Nachhinein betrachtet, möchte man auf die Erweiterungsmethoden nicht mehr verzichten, sie sind in meinen Augen sogar ein richtiges Highlight, ganz nach dem Motto: »Wenn man sie nicht kennt, braucht man sie nicht. Kennt man sie, möchte man sie nicht mehr missen.«
Mit der neuen Version 4.0 des .NET Frameworks versucht Microsoft nun eine Lücke zu anderen Technologien zu schließen. Bislang war der Zugriff auf dynamische Bibliotheken wie COM APIs nicht ganz einfach. Diese Lücke wird mit der Version 4.0 geschlossen. Die Programmierung beispielsweise von MS Office wird deutlich vereinfacht. Die schon in Kapitel 3, »Das Klassendesign«, behandelten optionalen Parameter gehören zu den Neuerungen, die im Zuge dessen in C# 4.0 Einzug gehalten haben.
Eine weitere wichtige Neuerung von C# 4.0 möchte ich Ihnen in diesem Kapitel zeigen. Es ist die Fähigkeit, nun auch das sogenannte späte Binden zu realisieren. Was ist darunter zu verstehen? Schauen wir uns dazu einmal das folgende Beispiel an:
class Program { static void Main(string[] args) { Mathematics math = new Mathematics(); long result = math.Addition(56, 88); } } class Mathematics { public long Addition(int x, int y) { return x + y; } }
Die Klasse Mathematics wird instanziiert – ein statischer Vorgang, der bereits zur Kompilierzeit durchgeführt wird.
Mithilfe der Reflection ließ sich ein solcher Aufruf auch früher schon dynamisch formulieren, wie das folgende Codefragment zeigt, das die Bekanntgabe des Namespace System.Reflection voraussetzt:
object math = new Mathematics(); Type mathType = math.GetType(); object obj = mathType.InvokeMember("Addition", BindingFlags.InvokeMethod, null, math, new object[] { 56, 88 }); long result = (long)obj;
Das Resultat des Konstruktoraufrufs von Mathematics wird einer Variablen vom Typ Object zugewiesen. Darauf besorgt man sich den Type des Objekts und ruft darauf die Methode InvokeMember auf. Unter Übergabe des Methodenbezeichners, des Elementtyps, der Referenz des abzurufenden Objekts und der Argumente für den Methodenaufruf wird ein Resultat gebildet, das am Ende nur noch in den Ergebnistyp long konvertiert werden muss.
In C# 4.0 können Sie dieses Coding unter Zuhilfenahme des neuen Schlüsselworts dynamic auch deutlich kürzer ausdrücken:
dynamic obj = new Mathematics(); long result = obj.Addition(56, 88);
Das dynamic-Schlüsselwort wird zur Kompilierzeit statisch geprüft. Die Methode Addition hingegen ist dynamisch und wird nicht zur Kompilierzeit geprüft. Natürlich werden Sie auch keine IntelliSense-Hilfe nach dem Punktoperator nutzen können, Sie müssen die Methode Addition manuell angeben.
Bei obj handelt es sich um ein dynamisches Objekt. Der Aufruf der Methode selbst ist dynamisch und wird erst zur Laufzeit geprüft.
7.12.1 Eine kurze Analyse 

Lassen Sie uns an dieser Stelle die Objektvariablen betrachten und diese direkt miteinander vergleichen. Sie kennen mit dem in diesem Abschnitt beschriebenen Schlüsselwort dynamic inzwischen drei Varianten:
- Object myObject = new Mathematics();
- var myObject = new { ... };
- dynamic myObject = new Mathematics();
Die zuerst aufgeführte Instanziierung deklariert eine Variable vom Typ Mathematics. Die Variable ist vom Typ System.Object. Der Code ist streng typisiert. Sie können der Variablen jedes Objekt zuweisen, vorausgesetzt, es ist vom Typ System.Object und bekannt.
Flexibler ist bereits der zweite Ausdruck. Auch hier liegt eine strenge Typisierung vor, aber der Typ muss erst zur Laufzeit gebildet werden. Typischerweise handelt es sich dabei um anonyme Typen, die zur Kompilierzeit gebildet werden.
Auch die letzte Variante mit dynamic wird sehr wohl statisch geprüft, aber der Aufruf auf der dynamic-Variablen erfolgt dynamisch. Zur Kompilierzeit steht noch nicht fest, welche Operationen mit dem Typ ausgeführt werden. Daher ist auch keine IntelliSense-Hilfe sichtbar.
7.12.2 Dynamische Objekte 

Mit der Einführung des Schlüssselworts dynamic wurden auch einige Klassen zum .NET Framework hinzugefügt, die auf dynamic aufsetzen. Diese Klassen befinden sich im Namespace System.Dynamic. Am interessantesten scheint hier die Klasse DynamicObject zu sein, mit der Klassen zur Laufzeit dynamisch erweitert werden können. Sie müssen die Klasse ableiten und können die abgeleitete Klasse zur Laufzeit um Objekteigenschaften erweitern und diese abrufen. Sehen wir uns das im folgenden Beispielprogramm an.
// ------------------------------------------ // Beispiel: ...\Kapitel 7\DynamicObjects // ------------------------------------------- using System; using System.Collections.Generic; using System.Dynamic; using System.Reflection; class Program { static void Main(string[] args) { dynamic pers = new Person(); pers.Name = "Peter"; pers.Alter = 12; pers.Ort = "Bonn"; pers.Telefon = 0181812345; Console.WriteLine("{0}, {1}, {2}, {3}", pers.Name, pers.Alter, pers.Ort, pers.Telefon); Console.ReadLine(); } } class Person : DynamicObject { Dictionary<string, Object> dic = new Dictionary<string, object>(); public string Name { get; set; } public int Alter { get; set; } public override bool TryGetMember(GetMemberBinder binder, out object result) { return dic.TryGetValue(binder.Name, out result); } public override bool TrySetMember(SetMemberBinder binder, object value) { dic[binder.Name] = value; return true; } }
Es ist die Klasse Person definiert, die die Klasse DynamicObject ableitet. Mit Name und Alter sind zwei Eigenschaften konkret festgelegt. Darüber hinaus enthält die Klasse Person ein Feld vom Typ der generischen Klasse Dictionary<>. Hierbei handelt es sich um eine Array-ähnliche Liste, in der alle Daten mit einem Schlüssel-Werte-Paar beschrieben werden. Um den Programmcode zu verstehen, ist eine genauere Kenntnis der Klasse Dictionary<> nicht notwendig; wir werden im nächsten Kapitel noch genauer darauf eingehen.
Das Dictionary<>-Objekt speichert Eigenschaften, die zur Laufzeit bestimmt werden können. In Main sind das zum Beispiel die beiden Eigenschaften Ort und Telefon eines Person-Objekts, das zuvor mit dynamic erstellt wird – eine Voraussetzung für alle Typen, die von DynamicObject abgeleitet sind.
Damit die dynamischen Eigenschaften sich auch in das Objekt eintragen können, sind die beiden geerbten Methoden TrySetMember und TryGetMember überschrieben. Beide weisen mit GetMemberBinder und SetMemberBinder sehr ähnliche erste Parameter auf, die das dynamische Member repräsentieren. Der Bezeichner der dynamischen Eigenschaft ist in der Eigenschaft Name der beiden Binding-Objekte zu finden.
Interessant werden dürfte die Klasse DynamicObject vermutlich in Zukunft im Zusammenhang mit Daten, deren Strukturen nicht vorhersehbar sind oder sich von Fall zu Fall ändern. An diesem Zusammenhang sei auf die Tabellen einer Datenbank erinnert. Wie einfach ließen sich die Felder durch dynamische Member eines DynamicObject-Objekts beschreiben?