Ruby on Rails Friday, March 28, 2014

If Engine were to inherit ActiveRecord, would that affect your decision to set a default instance?
In such a case, Car would be effectively coupled to both Engine and ActiveRecord.
Also, do you find Rails developers in general care about loose coupling?



On Friday, 28 March 2014 00:22:04 UTC+11, Dave Aronson wrote:
On Thu, Mar 27, 2014 at 3:48 AM, Matthew Riley
<mat...@matthewriley.name> wrote:

> Of the two examples below, which do you believe is more testable?
>
> Example 1:
>
> class Car
>  def initialise(engine)
>    @engine = engine
>  end
...
> Example 2:
>
> class Car
>  def initialise
>    @engine = Engine.new
>
>  end

Definitely #1, as it will accept any sort of engine-like thing you
care to put in it, so you can use a mock or stub or other kind of
fake, even without relying on a faking library to monkey with the item
(or worse yet, with Engine).

I might be tempted to make a slight change, though: declare the
initializer as "def initialise(engine = Engine.new)", so you don't
have to explicitly pass in an Engine in the default case.  Yes, that
re-couples them somewhat, that's the downside.  If you can be sure
that Car always has access to Engine, like if they're all part of the
same gem you're distributing, this is probably acceptable.  In fact,
you could do something like:

def initalize(options={})
  @engine = options[:engine] || Engine.new
  @transmission = options[:transmission] || Transmission.new
  @horn = options[:horn] || Horn.new
  # and so on
end

or if you want to really delve into funky Ruby, maybe even do some
metaprogramming to DRY up that pattern, like:

def initalize(options={})
  [:engine, :transmission, :horn].each do |sym|
    option = options[sym]
    klazz = Kernel.const_get(sym.capitalize)
    self.instance_variable_set(sym, option || klazz.new)
  end
end

That might be more worthwhile if you have a long list of options to pass in.

-Dave

--
Dave Aronson, freelance software developer (details @ www.Codosaur.us);
see also www.PullRequestRoulette.com, Blog.Codosaur.us, www.Dare2XL.com

--
You received this message because you are subscribed to the Google Groups "Ruby on Rails: Talk" group.
To unsubscribe from this group and stop receiving emails from it, send an email to rubyonrails-talk+unsubscribe@googlegroups.com.
To post to this group, send email to rubyonrails-talk@googlegroups.com.
To view this discussion on the web visit https://groups.google.com/d/msgid/rubyonrails-talk/61afeaf2-d543-4726-9d59-f6cbc0e2999b%40googlegroups.com.
For more options, visit https://groups.google.com/d/optout.

No comments:

Post a Comment