Foundations of Programming 2 - Chapter 3 - IoC 180
Jan 06, 2011
The book is being made available on github at the same time as here (for your contributing convenience). Please view for how you can help
When you first learn about IoC and start playing with a DI framework, you don't think to yourself I wonder how else I could resolve dependencies? Why would you? DI is straightforward as well as being a good fit with how most people organize their Java or .NET code. No one would blame you for thinking, like I did, that this is how everyone does it. It isn't. DI is a pattern suited for object oriented programming with static languages. Switch to a different paradigm and you'll find different solutions.
Dynamic Mind Shift
It's common for developers to think of dynamic languages and dynamic typing as synonyms. It's true that most (though not all) dynamic languages are dynamically typed. The fact though is that dynamic languages execute many things at runtime, which static languages do at compile time. The implication is that, within a dynamic runtime, developers have greater capabilities at runtime than their static counterparts.
At first such power might seem to be of limited use. After all, how often do you need to change your code at runtime? It seems though, like we often think a feature isn't useful until we have access to it. Consider the features which have been added to C#, before they existing, you possibly didn't even know they could exist. Eventually these additions reshaped how you coded: generics, anonymous methods, LINQ. A dynamic runtime is the same: the impact is difficult to grasp as long as the way you think is constrained by your experience with static languages. Reflection isn't an integral part of your work because it's both limited and cumbersome; yet make it powerful and simple and it might be a tool you leverage on a daily basis. (To be honest, comparing dynamic languages with reflection is hugely unjust to dynamic languages, we're just trying to draw some initial parallels.)
What does this have to do with Inversion of Control? The flexible nature of dynamic languages means that IoC is built directly into most dynamic languages. There's no need to inject parameters or use a framework, simply leverage the language's capabilities. Let's look at an example:
class User
def self.find_user(username, password)
user = first(:conditions => {:username => username})
user.nil? || !Encryptor.password_match(user, password) ? nil : user
end
end
Our code has two dependencies: first
will access an underlying store (which may not be obvious if you aren't familiar with Ruby) and Encryptor
is a made-up class responsible for matching a user's hashed password with a supplied password. In a static world both of these would be troublesome. In ruby, and other dynamic languages? Simply change what first
and Encryptor
do:
def password_match_returns(expected)
metaclass = class << Encryptor; self; end
metaclass.send :define_method, :password_match do
return expected
end
end
def first_returns(expected)
metaclass = class << User; self; end
metaclass.send :define_method, :first do
return expected
end
end
Keep an open mind and remember that this code may be as mysterious to you as anonymous methods and lambdas are to someone else. We'll discuss the code in further detail, but let's first look at how it might be used:
it "returns the found user" do
user = User.new
first_returns(user)
password_match_returns(true)
User.find_user('leto', 'ghanima').should == user
end
In real life you'd use a mocking framework to take care of this and provide a cleaner syntax and more powerful features. But, putting aside some of the magic, we can see that our two methods redefine the :first
and password_match
methods at runtime so that they implement a behavior that our test can use. To really start understanding this, we need to cover singleton classes.
Singleton and Metaclasses
In C#, Java and most static languages a class can safely be thought of as a rigid template. You define fields and methods and compile your code using classes as an immutable contract. Classes serve a very useful purpose as a design-time tool. The problem with classes, by no fault of their own, is that most programmers think classes and object-oriented programming are one and the same. They aren't. Object orientated programming is, as the name implies, about the living objects of your running code. In a static language this means instances. Since instances are tightly bound to the template defined by their respective class, it's easy to see why developers mix the two concepts.
Look beyond static languages though and you'll see a different story: not only is there no law that says classes cannot themselves be living things, but object oriented programming can happily exist without classes. The best example that you're probably already familiar would be from JavaScript. Behold, OOP without classes:
var leto = {
fullName: 'Leto Atreides II',
title: 'Emperor',
yearOfBirth: 10207,
getAngryWithDuncan: function(duncan) {
duncan.alive = false;
}
};
var duncan = {
ghola: true,
alive: true
};
leto.getAngryWithDuncan(duncan);
As always, the point isn't that one approach is better than another, but rather to gain a different perspective - likely, in this case, on knowledge you already possess. Doing a decade of object oriented programming a certain way is the kind of experience that can compromise your ability to grow.
So object oriented doesn't require classes, but as templates classes are quite handy. This is where ruby and singleton classes come in; because, as we've already mentioned, there's no law that says a class has to be a predefined and unchangeable template.
In ruby every object has its own class, called a singleton class. This let's you define members on specific instances, like:
class Sayan
end
goku = Sayan.new
vegeta = Sayan.new
def goku.is_over_9000?
true
end
p goku.is_over_9000? => true
p vegeta.is_over_9000? => NoMethodError: undefined method `is_over_9000?'
Technically, we aren't adding the is_over_9000?
method to the goku
object, we are adding it to its invisible singleton class, which goku
inherits from and thus has access to. We call the singleton class invisible because both goku.class
and vegeta.class
will return Sayan
. There are ways to expose a singleton class, but when you aren't doing metaprogramming, singleton classes are transparent.
To get access to a singleton class, which is itself a real object, we use the class <<
syntax. For example the is_over_9000?
method could alternatively be defined like so:
class << goku
def is_over_9000?
true
end
end
If we want to assign the singleton class to a variable, we can simply expose self
:
singleton = class << goku
self
end
singleton = class << goku; self; end
Interestingly (and I'm not too sure why I find it interesting), if we look at the goku
and singleton
instances, we get the following output:
goku
=>
singleton
=>
In Ruby, everything is an object. Even a class is an object (which inherits from Class
, which in turn inherits from Object
). That means you can invoke methods on your classes:
Sayan.is_a?(Object)
=> true
Saya.is_a?(Integer)
=> false
Sayan.to_s
=> "Sayan"
Since classes are objects they too have singleton classes (which are often called metaclasses). We can get access to a class' metaclass via the same class <<
syntax we used for an instance:
metaclass = class << Sayan
self
end
metaclass = class << Sayan; self; end
Singleton classes aren't really something you'll deal with too often, but metaclasses are important because class methods are defined in the metaclasses. Class methods are, in a lot of ways, like static methods in C# or Java. They are defined one of two ways and used like you'd expect:
class Sayan
def self.find_most_powerful()
end
class << self
def all_by_level(superSayanLevel)
end
end
end
powerfulSayans = Sayan.all_by_level(3)
(Understanding self
in Ruby, specifically what self
refers to in a given context, is important to mastering the language.)
The key difference though, between Ruby class methods and Java/C# static methods, is that class methods are defined against a metaclass which is an object. In other words, while class methods resemble static methods, they actually share more in common with instance methods.
What does all this get us? Much of the rigidness you'll bump up against in a static language doesn't exist in dynamic language. Sealed classes, non virtual methods and static methods, which are mechanisms to stop you from doing something, vanish. There are pros and cons to both approaches, but there's no reason not to be familiar with both.
I do want to point out that, from a testability perspective, metaprogramming does have significant advantages - the difficulty in testing a static password_match
method in C# should be proof enough of that. We can't simply overwrite the implementation, as we did at the start of this chapter in Ruby, because classes aren't objects. DI, or even interfaces, simply aren't necessary in Ruby. The decoupling you achieve in C# via injecting interfaces and managing dependencies is replaced by the very nature of the Ruby language.
Events/Callbacks
Another way to reduce coupling is to leverage events and callbacks. It's been long understood that events, by their very nature, provide protection against coupling. Code which raises an event is saying I don't care who you are or what you are going to do, but it's time to do it. There could be 0 listeners, or a hundred, they could do all sorts of unrelated things, but none of that matters to the calling code. The code ends up easy to test because, from the caller's point of view, you just need to verify that the event is raised and from the callee's point of view that they register for the event.
The reason we don't use events everywhere is because they just don't lend themselves to the linear flow we use for most code. However, over the last couple years, this has started to change. Why? The resurgence of JavaScript. We now have more code written in JavaScript and more developers are letting go of their old (generally negative) perceptions and learning the language anew. JavaScript is no longer a place where we can rely on hacks and hope it all keeps working. There's been a shift towards quality and maintainable JavaScript. That means we need to start worrying about coupling, cohesion and testability. When it comes to that, events are to JavaScript what DI is to Java/C# or metaprogramming is to Ruby.
Let's say you're a jQuery master (if you aren't, you can jump to Appendix A then B to start that journey whenever you want) and have built a series of generic plugins. These are things that we plan on reusing throughout our site. A lot of these plugins will need to interact with each other. For example, one of the plugins turns a simple list of rows into a pageable and sortable grid, and another allows a form to be submitted via ajax. Of course, when the user submits a new record, via the ajax form, the fancy grid needs to be updated. First, lets look at the basic setup:
$('#users').fancyGrid();
$('#add_user').fancySubmit();
The core of our fancySubmit
plugin will be something as simple as:
(function($)
{
$.fn.fancySubmit = function(options)
{
return this.each(function()
{
var $form = $(this);
var self =
{
initialize: function()
{
$form.submit(function()
{
$.post($form.attr('action'), $form.serialize(), self.handleResponse);
return false;
});
},
handleResponse: function(r)
{
}
};
this.fancySubmit = self;
self.initialize();
});
};
})(jQuery);
(If you aren't familiar with jQuery, this code is intercepting our form's submit
event in order to submit the data via ajax. The response from that ajax call is then handled by the handleResponse
function. Again, you might want to skip to Appendix A and B to get up to speed on jQuery.)
handleResponse
can handle generic cases (errors, validation) directly, but anything more specific will depend on the specific context it's being used in. The solution? Allow a callback to be passed into the plugin and trigger it when appropriate:
handleResponse: function(r){
if (options.onSubmit != null) { options.onSubmit(r); }
}
With that simple change, we can now tie the two plugins together, without really having to tie them together:
var $users = $('#users').fancyGrid();
$('#add_user').fancySubmit({
onSubmit: function(r) { $users.fancyGrid('newRow', r); }
});
Using this approach, the two plugins work together without knowing anything about each other. This helps ensure that your code stays highly reusable and cohesive.
Testing It
We can test handleResponse
by supplying a fake callback which can make some basic assertions. However, we have to deal with call to $.post
. Since JavaScript is dynamic like Ruby, it too provides a mechanism to change our runtime definitions. Combining our dynamic rewrite with our custom callbacks yields:
test("executes the onSubmit callback after receiving a response", function()
{
expect(1);
$.post = function(url, params, callback)
{
callback('the new row');
}
var $form = $('#a_test_form');
$form.fancySubmit(
{
onSubmit: function(r)
{
ok(r == 'the new row', 'callback with response was called');
}
});
$form.submit();
});
Ignore the call to expect
for now, we'll get to it in a minute. We rewrite the $.post
function, circumventing the heavy implementation with something controllable and lightweight. $.post
now essentially executes the 3rd parameter (which if we look back up at the plugin is a call to self.handleResponse
) with a hardcoded parameters. Next, we initialize the plugin with our callback, which will assert that the supplied parameter is what we expect. Finally, we actually submit the form to execute the actual code.
About the call to expect
, this tells our testing framework, Qunit in this case, that 1 expectation should be called. This is key when dealing with callbacks and events. What would happen if we didn't call expect
and also changed our plugin to not invoke our callback? Our test would still pass because nothing would ever be asserted. By specifying expect(1)
we ensure that our real expectation (that our callback is called, and called with the correct parameters) is invoked - if ok
isn't called, then expect
will fail and we'll know something isn't right.
The introduction of anonymous methods and lambdas make similar code possible, and even preferable in some situations, in C#.
In This Chapter
Depending on your experience with Ruby and jQuery, this chapter might have been overwhelming. We covered some advanced techniques in languages less familiar to us. We possibly picked the most complicated part of Ruby to look at (metaprogramming), so don't be discouraged if you missed some, or even most, of it. We also saw some pretty different testing scenarios. Most important though was how we were able to relearn something we thought we knew well (IoC) by learning what, to others, is pretty fundamental stuff. You can almost consider IoC a built-in feature of Ruby, much like you'd see LINQ as a built-in feature of C#. Even if you don't understand the nuance of the code or the implications it has on day to day programming, this chapter should still showcase the value of learning and growing.