require 'benchmark' class BlankSlate instance_methods.each { |meth| undef_method(meth) unless meth.to_s =~ /^(__|object_id)/ } end class NormalMessage < BlankSlate def initialize(*phrases) @message = ""; _phrase(*phrases) end def to_s; @message; end def method_missing(meth, *phrases, &blk) push(meth.to_s.gsub('_', ' ')); _phrase(*phrases); end def comma(str, *phrases) raise Exception, "Whoops - comma"; end def but(*phrases); comma("but", *phrases); end def not(*phrases); comma("not", *phrases); end def push(str) raise Exception, "Whoops - push"; end private def _concat(str) (@message << str).strip!; self; end def _phrase(*phrases) phrases.each { |phrase| push(phrase.inspect) }; self; end end # Message # + class PlusMessage < NormalMessage def comma(str, *phrases) _concat(", " + str); _phrase(*phrases); end def push(str) _concat(" " + str); end end # Message # << class AppendMessage < NormalMessage def comma(str, *phrases) _concat(", " << str); _phrase(*phrases); end def push(str) _concat(" " << str); end end # Message # Experimental class ExperimentalMessage < BlankSlate def initialize(*phrases) @chunks = []; _inspect(phrases) end def to_s; @chunks.join; end def method_missing(meth, *phrases, &blk) push(meth.to_s.gsub('_', ' ')); _inspect(phrases); end def comma(str, *phrases) _concat([", ", str]); _inspect(phrases); end def but(*phrases); comma("but", *phrases); end def not(*phrases); comma("not", *phrases); end def push(str) _concat([" ", str]); end private def _concat(chunks) @chunks.concat(chunks); self; end def _inspect(phrases) phrases.each { |phrase| push(phrase.inspect) }; self; end end # Message # # Benchmarking Benchmark.bmbm do |x| def message_test(klass) STDOUT.puts(klass.new.comma("yes").push("no").foo.bar("baz").to_s) 10_000.times do klass.new.comma("yes").push("no").foo.bar("baz").to_s end end x.report("+ based message") do message_test(PlusMessage) end x.report("<< based message") do message_test(AppendMessage) end x.report("experimental message") do message_test(ExperimentalMessage) end end