Lately I've been posting about using IL generation to connect elements.
I know I was about to post series about Scene Graph, but since I like to do things in an non ordered fashion (and at the end of the day, it's my blog ;) , I'll go a bit deeper into objects.
In most graph oriented architecture, we have to create nodes, and connectors, so we have to build inputs/outputs...
We have several ways to do it:
I know I was about to post series about Scene Graph, but since I like to do things in an non ordered fashion (and at the end of the day, it's my blog ;) , I'll go a bit deeper into objects.
In most graph oriented architecture, we have to create nodes, and connectors, so we have to build inputs/outputs...
We have several ways to do it:
- Big factory: You have calls like CreateIntParameter, CreateValueInput, or generic version like CreateInput<T> ...
- Injection : You use decorators in order to inject some implementation. (So doing ISpread<int> internally generates a IntInputPin or something like that).
Going deeper, why do we even need pins? Let's take the following example:
Code Snippet
- public class Prop
- {
- public double Hello { get; set; }
- public int Integer { get; set; }
- public string ReadOnly { get { return "ReadOnly"; } }
- }
- public class Prop2
- {
- public double Double { get; set; }
- public string DoubleOutput { get; set; }
- }
This is pretty easy to reflect properties, since we know about get/set, we can create a "virtual pin" automatically, so why even wrap the property?
In most systems we would work it like that:
So let's do this in a simpler way, let's say we want to connect Hello (output) to Prop2 "DoubleOutput"
First we need a Func to access Property:
And a way to set property:
Now we have to connect both, eg: read from source and write to destination:
That was that easy, now we can simply have:
All done, out connector is ready ;)
If we need conversion, this is as easy, we can use a small interface:
Then our connector can either use this small interface, or we can just set a lambda is case we can't be bother ed adding one file + namespaces + new class, get rid of object oriented verbosity basically ;)
In action:
Now we can actually simply reflect type, create IO on rules, but operate directly on properties.
Direct access costs us a few virtual calls, but since we haven't got a layer on top, this is even more minimized.
Once we have this, let's looks at it further, let's show how to display our element, with custom formatted string per type, in this example I just log to console, bu this is easily extendable to user interface, writing, serialization...
First we do some simple defaults:
Then two stupid int and double implementations:
Now we need to know if a property is capable of display of not:
Add filtering support:
From here we know from a given type, every property that we can display, and how to create them. We only used object type so far, so now we need to add instance in the mix:
And the final object that displays our information:
Our final code, that gets an instance and displays any supported data:
You can easily think that's a lot of classes and a lot of code to just display properties of a class (we could just iterate through property list and do some "tostring"
But...
In most systems we would work it like that:
- Wrap hello as input and Output Pin.
- Wrap ReadOnly as Output pin.
- Create a link that wraps InputPin<double> and OutputPin<double>
So let's do this in a simpler way, let's say we want to connect Hello (output) to Prop2 "DoubleOutput"
First we need a Func to access Property:
Code Snippet
- public static Func<T> BuildGetter<T>(object instance, PropertyInfo property)
- {
- return (Func<T>)Delegate.CreateDelegate(typeof(Func<T>), instance, property.GetGetMethod());
- }
And a way to set property:
Code Snippet
- static Action<T> BuildSetter<T>(object instance, PropertyInfo property)
- {
- return (Action<T>)Delegate.CreateDelegate(typeof(Action<T>), instance, property.GetSetMethod());
- }
Now we have to connect both, eg: read from source and write to destination:
Code Snippet
- static Action Connect<T>(object source, PropertyInfo getter, object dest, PropertyInfo setter)
- {
- var getFunc = BuildGetter<T>(source, getter);
- var setAction = BuildSetter<T>(dest, setter);
- return () => setAction(getFunc());
- }
That was that easy, now we can simply have:
Code Snippet
- Prop p = new Prop()
- {
- Hello = 20.0,
- Integer = 20
- };
- Prop2 p2 = new Prop2();
- var connect = Connect<double>(p, typeof(Prop).GetProperty("Hello"), p2, typeof(Prop2).GetProperty("Double"));
- connect();
All done, out connector is ready ;)
If we need conversion, this is as easy, we can use a small interface:
Code Snippet
- public interface ITypeConverter<TSource,TDest>
- {
- TDest Convert(TSource source);
- }
- public class DoubleToStringConverter : ITypeConverter<double, string>
- {
- public string Convert(double source)
- {
- return source.ToString() + " from converter";
- }
- }
Then our connector can either use this small interface, or we can just set a lambda is case we can't be bother ed adding one file + namespaces + new class, get rid of object oriented verbosity basically ;)
Code Snippet
- static Action Connect<TSource, TDest>(object source, PropertyInfo getter, object dest, PropertyInfo setter, Func<TSource,TDest> converter)
- {
- var getFunc = BuildGetter<TSource>(source, getter);
- var setAction = BuildSetter<TDest>(dest, setter);
- return () => setAction(converter(getFunc()));
- }
- static Action Connect<TSource, TDest>(object source, PropertyInfo getter, object dest, PropertyInfo setter, ITypeConverter<TSource, TDest> converter)
- {
- var getFunc = BuildGetter<TSource>(source, getter);
- var setAction = BuildSetter<TDest>(dest, setter);
- return () => setAction(converter.Convert(getFunc()));
- }
In action:
Code Snippet
- var connectConverter = Connect<double, string>(p, typeof(Prop).GetProperty("Hello"), p2, typeof(Prop2).GetProperty("DoubleOutput"), (d) => d.ToString() + " Hello");
- connectConverter();
- var connectInterface = Connect<double, string>(p, typeof(Prop).GetProperty("Hello"), p2, typeof(Prop2).GetProperty("DoubleOutput"), new DoubleToStringConverter());
- connectInterface();
Now we can actually simply reflect type, create IO on rules, but operate directly on properties.
Direct access costs us a few virtual calls, but since we haven't got a layer on top, this is even more minimized.
Once we have this, let's looks at it further, let's show how to display our element, with custom formatted string per type, in this example I just log to console, bu this is easily extendable to user interface, writing, serialization...
Code Snippet
- public interface IPropertyDisplayFactory
- {
- Type PropertyType { get; }
- IPropertyDisplay DisplayObject(object instance, PropertyInfo property);
- }
- public interface IPropertyDisplay
- {
- void Display();
- }
First we do some simple defaults:
Code Snippet
- public class BasicDisplay : IPropertyDisplay
- {
- private readonly Action display;
- public BasicDisplay(Action display)
- {
- this.display = display;
- }
- public void Display()
- {
- display();
- }
- }
- public abstract class TypedDisplayFactory<T> : IPropertyDisplayFactory
- {
- public Type PropertyType
- {
- get { return typeof(T); }
- }
- public abstract IPropertyDisplay DisplayObject(object instance, PropertyInfo property);
- }
Then two stupid int and double implementations:
Code Snippet
- public class DoubleDisplayFactory : TypedDisplayFactory<double>
- {
- public override IPropertyDisplay DisplayObject(object instance, PropertyInfo property)
- {
- Func<double> getter = Program.BuildGetter<double>(instance, property);
- Action action = () =>
- {
- Console.WriteLine(property.Name + " Double: " + getter().ToString());
- };
- return new BasicDisplay(action);
- }
- }
- public class IntDisplayFactory : TypedDisplayFactory<int>
- {
- public override IPropertyDisplay DisplayObject(object instance, PropertyInfo property)
- {
- Func<int> getter = Program.BuildGetter<int>(instance, property);
- Action action = () =>
- {
- Console.WriteLine(property.Name + "Int: " + getter().ToString());
- };
- return new BasicDisplay(action);
- }
- }
Now we need to know if a property is capable of display of not:
Code Snippet
- public class DisplayPropertyRegistry
- {
- private readonly IEnumerable<IPropertyDisplayFactory> factories;
- public DisplayPropertyRegistry(IEnumerable<IPropertyDisplayFactory> factories)
- {
- this.factories = factories;
- }
- public DisplayPropertyRegistry(params IPropertyDisplayFactory[] factories)
- {
- this.factories = factories;
- }
- private IEnumerable<IPropertyDisplayFactory> SearchCandidate(Type type)
- {
- return factories.Where(f => f.PropertyType == type);
- }
- public bool CanDisplay(Type type)
- {
- return SearchCandidate(type).FirstOrDefault() != null;
- }
- public IPropertyDisplayFactory GetFactory(Type type)
- {
- return SearchCandidate(type).First();
- }
- }
Add filtering support:
Code Snippet
- public class DisplayPropertyFilter
- {
- private readonly DisplayPropertyRegistry registry;
- public DisplayPropertyFilter(DisplayPropertyRegistry registry)
- {
- this.registry = registry;
- }
- public IEnumerable<Tuple<PropertyInfo,IPropertyDisplayFactory>> ReflectType(Type type)
- {
- var props = type.GetProperties().Where(p => p.GetGetMethod() != null && registry.CanDisplay(p.PropertyType));
- return props.Select(p => newTuple<PropertyInfo,IPropertyDisplayFactory>(p, registry.GetFactory(p.PropertyType)));
- }
- }
From here we know from a given type, every property that we can display, and how to create them. We only used object type so far, so now we need to add instance in the mix:
Code Snippet
- public class DisplayObjectFactory
- {
- private readonly DisplayPropertyFilter filter;
- public DisplayObjectFactory(DisplayPropertyFilter filter)
- {
- this.filter = filter;
- }
- public ObjectDisplay GetDisplay(object instance)
- {
- var factories = filter.ReflectType(instance.GetType());
- var elements = factories.Select(f => f.Item2.DisplayObject(instance, f.Item1));
- return new ObjectDisplay(elements);
- }
- }
And the final object that displays our information:
Code Snippet
- public class ObjectDisplay
- {
- private readonly IEnumerable<IPropertyDisplay> displayElements;
- public ObjectDisplay(IEnumerable<IPropertyDisplay> displayElements)
- {
- this.displayElements = displayElements;
- }
- public void Display()
- {
- foreach (IPropertyDisplay display in displayElements)
- {
- display.Display();
- }
- }
- }
Our final code, that gets an instance and displays any supported data:
Code Snippet
- DisplayPropertyRegistry registry = new DisplayPropertyRegistry(
- new DoubleDisplayFactory(),
- new IntDisplayFactory());
- DisplayPropertyFilter filter = new DisplayPropertyFilter(registry);
- DisplayObjectFactory builder = new DisplayObjectFactory(filter);
- Prop p = new Prop()
- {
- Hello = 20.0,
- Integer = 20
- };
- var display = builder.GetDisplay(p);
- display.Display();
You can easily think that's a lot of classes and a lot of code to just display properties of a class (we could just iterate through property list and do some "tostring"
But...
- This is more extensible, and if you need to access your properties a lot, you get a pretty big gain over reflection.
- You can register any type, replace as you wish, so you avoid the dreaded massive switch.
- By having more classes, and less work per class, you have much more invariants, pretty much every class is ready to go once constructed, no temporal coupling (only edge case is CanDisplay/GetFactory which could create a related exception).
- You are not limited to just a "tostring", create a controller, network serializer, add a proxy for automatic property change dispatch, possibilities are endless ;)
Here we are for now, promised I'll get back into Scene Graph next post (or maybe not ;)
No comments:
Post a Comment