I've been thinking a lot about intentional programming and the idea that behavior doesn't simply belong to an entity but to the collaboration of two entities and the surrounding context in which they find themselves.
It's an old pattern in database and data modeling to represent rich relationships between two entities with three entities rather than two, where the third entity contains the new information created by the relationship between the first two.
Dependency Injection is a mechanism for permitting this same basic pattern for OO programming languages, where the third entity is really the larger enclosing context in which both of the primary entities exist.
I'm chewing on an intersection of two interesting ideas that elaborate on this concept.
The first idea was inspired by reading maverick physicist Julian Barbour's book "The End of Time: The Next Generation in Physics." One of Barbour's pet conceits is that there is no causality. That all moments in time exist simultaneously as bubbles in a higher order dimension and that causality is really the apparent ordering which is evident when you look in any direction across the sea of bubbles.
This notion got me thinking about entities being nested inside of other entities and the whole notion of hierarchical scope being turtles all the way down. Infinite hierarchical inclusion is a comfortingly attractive idea but in practice it proves a stumbling block when trying to cope with cross-tree relationships. Instead of having a fundamental notion of all entities being defined somewhere in a single-footed hierarchy which owns them, what if we instead said that all entities are their own single-node trees, living in their own world and that their presence in any other larger "enclosing" context was by relationship.
That gives us a model where container->contained relationships are not special cases. They are the normal case. And A->B relationships might be modeled as A->container->B. This is vaguely less efficient than simple A->B, but vastly more powerful because concepts like friction, physics and indirect coupling can now be modeled explicitly.
It also meshes nicely with my second thought, which is that instead of an OOP-driven point of view where behaviors belong to entities, it seems more rational to think of behaviors as belonging to relationships. So that what is possible for entity A to do or have done to it within context C, is a function of the relationship between A and C. Likewise, what A can do to B within enclosing context C is a function of the relationship of A->C and of C->B.
I'll post the math for this when I get further along in working it out. It seems to me though that the confluence of these two simple thoughts is a powerful basis for a language that permits entities to be used differently in different contexts with higher levels of reuse and repurposing than is found in other systems.
1 comment:
Hi Mack,
I like your thinking on this. There must be something about intentional programming that leads to this conclusion, because I reached the same mode of thinking working on a project that was very similar to intentional programming.
The product was a graphical development tool for creating firmware in the problem domain using a unique graphical language.
It got me to thinking about the end product and how I might use similar ideas in the design of code for that product. I ended up in frustration over the lack of supporting features in the language I was using (C#). I ended up coding a lot of container-specific methods on the contained object interfaces as well as methods for use by other contained objects but only through the container. Exactly what you more eloquently described.
It would be nice if OOP languages provided a way to limit the interface usage in that way without having to resort to method-naming conventions to remind the programmer of the context.
Great minds think alike, I guess.
Post a Comment