MultidispatchDSL
If you worked with other languages you probably know multi dispatch feature - define variants of same method with different number of arguments of different types. In Ruby with it’s duck typing you cannot do it from out of box. This gem will allow you to do it.
<img src=“
” /> <img src=“travis-ci.org/jalkoby/uni_sender_gem.png?branch=master” alt=“Build Status” /> <img src=“
” alt=“Gem Version” />
Installation
Add this line to your application’s Gemfile:
gem 'multidispatch_dsl'
And then execute:
$ bundle
Or install it yourself as:
$ gem install multidispatch_dsl
Usage
Just include MultidispatchDSL into your class and your will get mdef class method. For example
class TestClass
include MultidispatchDSL
# hello method with one integer argument
mdef(:hello, Fixnum) do |i|
"Fixnum version with number #{ i }"
end
# hello method with one string argument
mdef(:hello, String) do |str|
"String version with string #{ str }"
end
# hello method with fixnum and string arguments
mdef(:hello, Fixnum, String) do |i, str|
"Fixnum String version with #{ i } & #{ str }"
end
# hello method with string and fixnum arguments
mdef(:hello, String, Fixnum) do |str, i|
"String Fixnum version with #{ str } & #{ i }"
end
# hello method without arguments
mdef(:hello) do
"Version without args"
end
# hello method with symbol argument
mdef(:hello, Symbol) do |symbol|
"Symbol version with :#{ symbol } & :#{ internal_method }"
end
# hello method with 2 string argument and nested block
mdef(:hello, String, String) do |str_one, str_two, &block|
instance_exec(str_one.upcase, str_two.downcase, &block)
end
def internal_method
:internal_method
end
end
test_instance = TestClass.new
test_instance.hello
#=> "Version without args"
test_instance.hello(1)
#=> "Fixnum version with number 1"
test_instance.hello(:foo)
#=> "Symbol version with :foo & :internal_method"
test_instance.hello(1, "string")
#=> "Fixnum String version with 1 & string"
test_instance.hello("string", 1)
#=> "String Fixnum version with string & 1"
test_instance.hello(:not, :defined)
#=> raise error MultidispatchDSL::MissingDeclarationError
test_instance.hello('One', 'Two') { |str_one, str_two| "#{str_one} #{ str_two } #{ internal_method }" }
#=> "ONE two internal_method"
As you can see defining version of method with yield is little bit tricky. That because defining method from block add extra scope. More details about it here www.andylindeman.com/2011/01/08/defining-methods-using-blocks-in-ruby.html
If you need define version of method for any number arguments of any types use :anything symbol
mdef(:process, String) do |value|
{ :string => value }
end
mdef(:process, Fixnum) do |value|
{ :int => value }
end
mdef(:process, :anything) do |value|
{ :anything => value }
end
If you desire add this functionality to all classes just include MultidispatchDSL to Object class:
Object.send(:include, MultidispatchDSL)
Requirements
Ruby >= 1.9.2 and other ruby implementations with 1.9 mode
Contributing
-
Fork it
-
Create your feature branch (‘git checkout -b my-new-feature`)
-
Commit your changes (‘git commit -am ’Add some feature’‘)
-
Push to the branch (‘git push origin my-new-feature`)
-
Create new Pull Request