DEV Community

Brandon Weaver
Brandon Weaver

Posted on • Originally published at Medium on

Ruby 2.7 — Enumerable#tally

Ruby 2.7 — Enumerable#tally

Christmas has come and passed, 2.6 has been released, and now it’s time to mercilessly hawk the releases page for 2.7 features so we can start our fun little annual tradition of blogging about upcoming features.

Typically this means another December release, but there have been cases of methods making it in earlier if they’re merged to trunk this early in the year.

This round? We have the new method Enumerable#tally!

The Short Version

tally counts things:

[1, 1, 2].tally
# => { 1 => 2, 2 => 1 }

[1, 1, 2].map(&:even?).tally
# => { false => 2, true => 1 }
Enter fullscreen mode Exit fullscreen mode

Examples

The example used in Ruby’s official test code is:

[1, 2, 2, 3].tally
# => { 1 => 1, 2 => 2, 3 => 1 }
Enter fullscreen mode Exit fullscreen mode

Without a block, tally works by counting the occurrences of each element in an Enumerable type. If we apply that to a list of another type it might be a bit clearer:

%w(foo foo bar foo baz foo).tally
=> {"foo" => 4, "bar" => 1, "baz" => 1}
Enter fullscreen mode Exit fullscreen mode

Currently tally_by has not been accepted into core, so to tally by a function you would instead use map first:

%w(foo foo bar foo baz foo).map { |s| s[0] }.tally
=> {f => 4, b => 2}
Enter fullscreen mode Exit fullscreen mode

There’s discussion happening at the moment on accepting this feature, which would make the above syntax:

%w(foo foo bar foo baz foo).tally\_by { |s| s[0] }
=> {f => 4, b => 2}
Enter fullscreen mode Exit fullscreen mode

Why Use It?

If you’ve been using Ruby, chances are you’ve used code something like one of these lines to do the same thing tally is doing above:

list.group_by { |v| v.something }.transform_values(&:size)

list.group_by { |v| v.something }.map { |k, vs| [k, vs.size] }.to_h

list.group_by { |v| v.something }.to_h { |k, vs| [k, vs.size] }

list.each_with_object(Hash.new(0)) { |v, h| h[v.something] += 1 }
Enter fullscreen mode Exit fullscreen mode

There are likely several more variants of this, but those are a few of the more common ones you might see around. This is a nicety method to abbreviate a very common idiom in the Ruby language, and a very welcome one.

Vanilla Ruby Equivalent

What does this method do? Well if we were to implement it in plain Ruby it might look a bit like this:

module Enumerable
  def tally_by(&function)
    function ||= -> v { v }

    each_with_object(Hash.new(0)) do |value, hash|
      hash[function.call(value)] += 1
    end
  end

  def tally
    tally_by(&:itself)
  end
end
Enter fullscreen mode Exit fullscreen mode

In the case of no provided function, it would effectively be tallying by itself, or rather an identity function.

An identity function is a function that returns what it was given. If you give it 1, it returns 1. If you give it true, it returns true. Ruby also uses this concept in a method called itself.

This article will not go into great depth on what the above code does. Part Five of “Reducing Enumerable” covers this code in more detail:

Reducing Enumerable — Part Five: Cerulean, Master of Tally By

The Source Code

Nobu recently committed a patch to Ruby core to add this method:

enum.c: Enumerable#tally · ruby/ruby@673dc51

It was accepted by the Ruby core team under the name tally:

Feature #11076: Enumerable method count_by - Ruby trunk - Ruby Issue Tracking System

Tally?

Let’s start with what the word means:

A tally is a record of amounts or numbers which you keep changing and adding to as the activity which affects it progresses.

Now where did this name come from? Originally the name count_by was proposed, but the name was rejected as it differed from count which has a different return type and behavior.

On a car ride back from the Tahoe area and RailsCamp West we (myself, David, Stephanie, and Shannon) were discussing potentially alternate names to try and propose to see if the feature could get in under a different name.

David had proposed tally, and formally suggested it. It looks like the name stuck, and the code’s been merged into trunk.

Now I’d presented a talk at a few conferences, and decided to mention tally_by instead of count_by in my RubyConf talk in one section. The written version is over here:

Reducing Enumerable — Part Five: Cerulean, Master of Tally By

Just a bit of interesting backstory.

Wrapping Up

2.7 is on its way, let’s see what it’ll bring! I’m looking forward to seeing where Ruby goes from here.

Top comments (0)