![]() |
Typen / Klassen als Parameter verwenden
Moin,
in Smalltalk kann ich sowas machen:
Code:
(Ja, es geht hierbei um Klassen, nicht Instanzen / Exemplare von Klassen).
classes := {ClassA. ClassB. ClassC}.
classes do: [:each | each register] In Delphi sollte derartiges ja über das class of Konstrukt auch möglich sein. Ich bräuchte allerdings ein praktikables Äquivalent in C# und da konnte ich bis jetzt nichts finden. Ein weiterer netter Anwendungsfall wäre dann
Code:
void foo(ASuperClass c) {
bar = new c() } |
AW: Typen / Klassen als Parameter verwenden
Ein solches Konstrukt gibt es in C# nicht. Habe ich auch schon vergeblich nach gesucht :?
Es ließe sich aber evtl. durch Generics lösen:
Code:
Dann muss die übergebene Klasse aber auch einen Konstruktor mit 0 Argumenten besitzen.
void foo<T>() where T: BasisKlasse, new()
{ bar = new T(); } // ... foo<BasisKlasse>(); foo<AbgeleiteteKlasse>(); |
AW: Typen / Klassen als Parameter verwenden
Zitat:
für das zweite Beispiel sollte das mit den Generics ausreichend sein, wenn es klappt. Für den ersten hilft das allerdings auch nichts :stupid: |
AW: Typen / Klassen als Parameter verwenden
Schau dir mal den Namespace System.Type an, und den Operator typeof().
|
AW: Typen / Klassen als Parameter verwenden
Heißt das Namespace nicht System.Reflection? :stupid:
Das ist aber eigentlich auch keine Entsprechung für class-of, sondern für die RTTI... Aber damit sollte sich das erste Beispiel bewältigen lassen. |
AW: Typen / Klassen als Parameter verwenden
Zitat:
Code:
Aber wie komme ich nun vom Type wieder zur Klasse? Ein Eintrag aus dem Array kennt ja nun logischerweise die (statischen) Methoden meiner Klassen nicht (ist ja auch vom Typ Type) und ein cast ist nicht möglich :(
Type[] KnownClasses = new Type[] { typeof(ClassA), typeof(ClassB) };
Code:
foreach (Type t in KnownClasses) {
// ? t.foo() } |
AW: Typen / Klassen als Parameter verwenden
Die Lösung ist eine Methode von System.Type, nämlich InvokeMember :wink:
|
AW: Typen / Klassen als Parameter verwenden
Ich denke, ein konkretes Anwendungsbeispiel wäre langsam hilfreich :) . Von Dependency Injection bis zu Funktionaler Programmierung (Higher-Order-Functions) gibt es manchmal sogar bessere Lösungen als virtuelle statische Methoden.
Eine virtuelle statische Methode direkt umzusetzen, ohne Verlust von statischer Typisierung, wirst du aber nur erreichen, indem du sie zur Instanzmethode machst und dafür ggf. in eine eigene Klasse auslagerst. |
AW: Typen / Klassen als Parameter verwenden
Zitat:
Daher ist das erste Beispiel sowieso Schwachsinn. Da müsste Microsoft erstmal Delphi's Klassenmethoden abgucken.:lol: |
AW: Typen / Klassen als Parameter verwenden
Zitat:
Code:
Alle bekannten Events sollen nun bei einem Dispatcher registriert werden (der ServerEvent.Command aus den TCP Daten liest und dann ServerEvent.handler() passend aufruft).
abstract class ServerEvent
{ public static string command() { return ""; } public static void handler() { } } Mit der im Eingangspost gezeigten Smalltalk-Variante wäre ich da voll zufrieden :stupid: Vermutlich gibt es dafür auch völlig andere Lösungsansätze; sollte meine Idee tatsächlich nur schwer umsetzbar sein, werde ich wohl auch umstellen... |
AW: Typen / Klassen als Parameter verwenden
Nicht so kompliziert denken und einfach Instanzmethoden daraus machen ;) ? Ich sehe da keinen Nachteil - und wie gesagt können sich dadurch sogar ganz andere Möglichkeiten eröffnen :) :
Code:
Wirf eine weitere Assembly mit einer ServerEvent-Ableitung in den bin-Ordner und MEF wird sie ohne weiteres Zutun ebenfalls dem Dispatcher zuordnen. Und jetzt sag mir, dass das mit statischen virtuellen Methoden eleganter lösbar sei :) .
class Dispatcher
{ [ImportMany] IEnumerable<ServerEvent> EventHandlers { get; set; } public Dispatcher() { // Here be MEF magic... var catalog = new DirectoryCatalog(@".\"); var container = new CompositionContainer(catalog); container.Composeparts(this); } void Receive() { ServerEvent handler = EventHandlers.First(h => h.Command == command); handler.Handle(); } } public abstract class ServerEvent { public virtual string Command { get { return ""; } } public void Handle(); } [Export(ServerEvent)] class IAmATeapotServerEvent : ServerEvent { ... } |
AW: Typen / Klassen als Parameter verwenden
Das klingt natürlich wirklich nett. Einen Haken hat das ganze allerdings: das gibts alles erst ab .Net 4 oder? Ich bin aber an 3.5 gebunden, durch andere Libraries...
Zitat:
|
AW: Typen / Klassen als Parameter verwenden
Du kannst doch einfach per reflection hergehen, die statischen Methode ermitteln und aufrufen.
Um die Typen zu finden die diese Methode implementieren kannst du im Prinzip hergehen und mittels linq abfragen, welche typen eine statische Methode dieses namens haben. Das ist aber nicht gerade Performant, weil die Abfrage alle typen nach der Methode abfragt. |
AW: Typen / Klassen als Parameter verwenden
Zitat:
Mal gucken ob ich nicht doch auf 4 umsteigen kann... |
AW: Typen / Klassen als Parameter verwenden
Hast Du Einfluss auf die Klassen die Du da verwendest? Kannst Du die z.B. mit einem Custom Attribut versehen oder von einem bestimmten Interface ableiten (ich weiss, bei statischen Methoden ist das eher Sinnlos, aber z.B. INamingContainer ist auch nur ein flagging interface das keine Methoden oder Properties vorgibt).
Das macht dann das Ermitteln der entsprechenden Typen ziemlich einfach. Ps: Ich erwähne jetzt nur am Rande das Prism virtuelle statische Methoden schon länger unterstützt: ![]() |
AW: Typen / Klassen als Parameter verwenden
Zitat:
Wie sich das nun auswirkt, verstehe ich allerdings nicht, denn von Reflection in C# habe ich ehrlichgesagt null Ahnung ;) |
AW: Typen / Klassen als Parameter verwenden
Du kannst sowas machen:
Code:
Du hast in results dann eine Liste aller Typen, die Dein Flagging interface haben.
var results = from type in someAssembly.GetTypes()
where typeof(IFoo).IsAssignableFrom(type) select type; Der Aufruf sieht dann in etwa so aus:
Code:
var t := results.First();
foreach (MethodInfo mi in t.GetMethods(BindingFlags.Static | BindingFlags.Public)) { Console.WriteLine("method {0}", mi.Name); mi.Invoke(null, null); // parameter entsprechend setzen. } |
AW: Typen / Klassen als Parameter verwenden
Zitat:
![]()
Code:
ersetzen.
EventHandlers = Assembly.GetEntryAssembly().GetTypes().Where(t => t.IsSubclassOf(typeof(ServerEvent))).ToList();
Zitat:
Zitat:
Und ohne dir nahe treten zu wollen, oder überhaupt zu wissen, wie tief du schon in der CLR drinsteckst, ist es imo ein beeinträchtigender Fehler von Umsteigern, in einer managed Sprache Instanzierungen zählen zu wollen. Sagt mir jedenfalls meine eigene Erfahrung - man muss loslassen können, um zu genießen :stupid: . @Sebastian: Uh, jetzt musst du mir aber dringend erklären, wo du nun noch einen Vorteil von statischen Methoden + Reflection siehst. |
AW: Typen / Klassen als Parameter verwenden
Zitat:
|
AW: Typen / Klassen als Parameter verwenden
Zitat:
Code:
wird bei mir nicht zugelassen mit Fehler Verwendung eines Typen wie Variable etc... Ich hab ees dann hauptsächlich so versucht:
[Export(ServerEvent)]
Code:
da wird dann allerdings außer der abstrakten Basisklasse garnichts importiert. So wurde das ganze allerdings in den Beispielcodes die ich gefunden habe angewendet :gruebel:
[Export(typeof(ServerEvent))]
[ImportMany(typeof(ServerEvent))] Ich bitte um Erleuchtung :stupid: |
AW: Typen / Klassen als Parameter verwenden
Das
Code:
muss auch bei allen anderen Events sein, auch wenn es sich um Kindklassen von ServerEvent handelt:
[Export(typeof(ServerEvent))]
Code:
// wird exportiert ...
[Export(typeof(ServerEvent))] class ServerEvent { .. } // wird ebenfalls exportiert ... [Export(typeof(ServerEvent))] class DummyEvent : ServerEvent { ... } // wird nicht exportiert ... // trotz der Vererbung von ServerEvent class WuppdiEvent : ServerEvent { } |
AW: Typen / Klassen als Parameter verwenden
Zitat:
|
AW: Typen / Klassen als Parameter verwenden
Da rächt es sich, dass ich MEF noch nie verwendet habe :stupid: .
Code:
funktioniert. Vielleicht ignoriert der DirectoryCatalog Exe-Assemblies.
var catalog = new AssemblyCatalog(System.Reflection.Assembly.GetExecutingAssembly());
(Die nächsten 10 Tage im Urlaub...) |
AW: Typen / Klassen als Parameter verwenden
Zitat:
Deine neue Lösung funktioniert jedenfalls wunderbar :thumb: |
Alle Zeitangaben in WEZ +1. Es ist jetzt 08:54 Uhr. |
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024-2025 by Thomas Breitkreuz