I recently ran a small workshop about software design patterns, and I had a few examples for some of the most common patterns. I felt pretty comfortable with the presentation overall, especially where I was able to use a blog post by Sandi Metz to illustrate the Strategy Pattern. Today I gave a presentation about code katas: what they are, how they are useful, when you might go over a kata, and showed a few examples. One of the katas that caught my eye today, from CodeWars, was about determining if a given number is prime.
Sandi's example about the rhyme was great, and I'm not going to attempt to improve upon it. However, using something as simple as whether a number is prime can help make the underlying patterns all the more obvious. By ridiculously over-engineering something simple, the mechanisms are laid bare and can then be transferred to more realistic, complicated situations. I'll be using the excuse of discovering if a given number is prime to explore the Strategy pattern.
I created a simple method of discovering if a number is a prime. I just perform a series of checks using guard clauses. For example, since 2 is prime, I have a check to return true for that input, and since a prime number has to be greater than 1, if an input is less than 2, I return false. There are then other checks on what is not a prime before we just return true, since the input must be a prime at that point.
def is_prime?(input)
return true if input == 2
return false if input < 2
return false if input.even?
sqrt = Math.sqrt(input)
return false if sqrt == sqrt.floor
(3..(sqrt.floor)).each do |i|
return false if (input%i).zero?
end
return true
end
I used TDD to build even this simple solution. The spec file is straightforward, and has a lot of room for improvement to DRY it up, but that can be an exercise for a different article.
With the naive solution now available, we can move those rules into individual strategies to be processed sequentially. This will allow us to update the strategies or add new ones independently from the rest of the code.
Looking over the list of steps, something stands out. Each guard clause consists of a return false if
statement, with a final return true
to mark a number as prime after we've exhausted the checks for numbers that aren't prime.
Except, that is, for the first check on whether the input is 2. If we remove that, a spec fails
context "for input of 2" do
let(:input) { 2 }
it { is_expected.to be_truthy }
end
Which makes sense. We return false if the input is even: return false if input.even?
We just change that, and the spec passes. So now we have the naive solution of
def is_prime?(input)
return false if input < 2
return false if input > 2 && input.even?
sqrt = Math.sqrt(input)
return false if sqrt == sqrt.floor
(3..(sqrt.floor)).each do |i|
return false if (input%i).zero?
end
return true
end
Everything is consistent now, we return false if a given check is true, otherwise we move on to the next check. I'm not comfortable returning false if a check is true, it could be confusing to a developer coming along in the future, but I'll shelve that concern for now. We can perhaps make this a little more obvious later with appropriate naming.
We can start by moving the first check into a strategy, and calling it directly.
module Strategies
class LessThanTwo
def check(number)
return number < 2
end
end
end
def is_prime?(input)
return false if Strategies::LessThanTwo.new.check(input)
Since I had used tests to check the original version, it's easy to verify that this change hasn't messed anything up by running those same tests. Unless there is some auto-loading going on, there will be a failure until require 'strategies'
is added to the top of the file. Since it's quick, I'll move the next check into a separate strategy. Now the code is
module Strategies
class LessThanTwo
def check(number)
return number < 2
end
end
class IsEven
def check(number)
return number > 2 && number.even?
end
end
end
I created a new transitional file to make these changes so they can be compared without having to dig through the history.
require 'strategies'
class PrimeTransition
def is_prime?(input)
return false if Strategies::LessThanTwo.new.check(input)
return false if Strategies::IsEven.new.check(input)
sqrt = Math.sqrt(input)
return false if sqrt == sqrt.floor
(3..(sqrt.floor)).each do |i|
return false if (input%i).zero?
end
return true
end
end
The next two checks aren't much more difficult than the last two, just moving each block of lines into a new strategy and converting the check into the familiar format of calling the new strategy.
return false if Strategies::HasIntegerSquareRoot.new.check(input)
return false if Strategies::HasDivisor.new.check(input)
class HasIntegerSquareRoot
def check(number)
sqrt = Math.sqrt(number)
return sqrt == sqrt.floor
end
end
class HasDivisor
def check(number)
sqrt = Math.sqrt(number).floor
(3..sqrt).each do |i|
return true if (number%i).zero?
end
return false
end
end
All specs pass, but the last strategy is a little different. It returns true in the middle of the loop, then finishes up with returning false, where the other strategies just return a conditional.
One way to make that strategy more consistent, and possibly more elegant, is by using any?
instead of each
. The any?
method passes each entry into the block to be tested however you wish, then returns true
if at least one satisfies the conditional is true, or else returns false
if every element fails the conditional.
class HasDivisor
def check(number)
sqrt = Math.sqrt(number).floor
return (3...sqrt).any? do |i|
(number%i).zero?
end
end
end
We now have a tidy method, with all logic packaged up in discrete methods that are completely independent.
def is_prime?(input)
return false if Strategies::LessThanTwo.new.check(input)
return false if Strategies::IsEven.new.check(input)
return false if Strategies::HasIntegerSquareRoot.new.check(input)
return false if Strategies::HasDivisor.new.check(input)
return true
end
We won't have to touch the is_prime?
method, or any of the other strategies, if one of the strategies is later found to be somehow lacking. However, this isn't quite the "Strategy Pattern". What if we need to add a strategy? Or find we can remove one, for whatever reason? We would have to change this file, and add a new strategy to lib/strategies.rb
(or wherever the strategies live). This isn't a big deal right now, just modifying two files, but if we were in actual code, we might be using these strategies in numerous places. Having them hardcoded each time would be inconvenient to maintain.
Instead, what if we passed these strategies in? In our contrived example, we can have is_prime?
call a new subordinate method and pass in the strategies that way.
class PrimeByStrategy
def is_prime?(input)
strategies = [Strategies::LessThanTwo.new,
Strategies::IsEven.new,
Strategies::HasIntegerSquareRoot.new,
Strategies::HasDivisor.new]
return is_prime_by_strategy?(input, strategies)
end
def is_prime_by_strategy?(number, strategies = [])
return false if strategies.any?{|s| s.check(number)}
return true
end
end
I'm still not happy with assembling the strategies right here. We have the logic of each of the strategies in the Strategies module, why not move the list of prime strategies there, too?
module Strategies
PRIME_STRATEGIES = [Strategies::LessThanTwo,
Strategies::IsEven,
Strategies::HasIntegerSquareRoot,
Strategies::HasDivisor]
And then we can just call on that, and if we add or remove strategies later, our PrimeByStrategy
doesn't care.
class PrimeByStrategy
def is_prime?(input)
return is_prime_by_strategy?(input, Strategies::PRIME_STRATEGIES.map(&:new))
end
def is_prime_by_strategy?(number, strategies = [])
return false if strategies.any?{|s| s.check(number)}
return true
end
end
This is looking better, but it seems a little odd to have that one line in the is_prime?
method. We can push those methods back together again, and make the PRIME_STRATEGIES
constant the default for the method.
class PrimeByStrategy
def is_prime?(input, strategies = Strategies::PRIME_STRATEGIES.map(&:new))
return false if strategies.any?{|s| s.check(input)}
return true
end
end
At this point, we're done showing off the strategy pattern. There is just one little thing that has been bothering me. We call a method called check
on the strategies, but get back a boolean. In Ruby, we can mark methods that pass back booleans by putting a question mark on the end. We could drop a ?
on there and leave it as check?
, but that is still rather vague. How about a method called not_prime?
?
class LessThanTwo
def not_prime?(number)
return number < 2
end
end
And call it from is_prime?
def is_prime?(input, strategies = Strategies::PRIME_STRATEGIES.map(&:new))
return false if strategies.any?{|s| s.not_prime?(input)}
return true
end
By naming the method not_prime?
we indicate that we will be returning a boolean here, and it is more descriptive than the vague check
, helping remedy my concern from earlier about returning false if a given check is true
. It is more reasonable now that we would return false
if the method not_prime?
is true.
We could keep going on other points, such as why I made each strategy a class that is instantiated instead of just using class methods that would be called directly, but perhaps I can go over that in another article.