Skip to content

Literal Ruby types

In Ruby, a type is any object that responds to === (triple-equals case equality) with a truthy or falsy value. Pretty much every object in Ruby is already a type.

For example, Ruby classes respond to === by checking if the given object is an instance of the class or an instance of a subclass of the class.

ruby
String === "Hello" # => true
String === 42 # => false

Objects typically respond to === by checking if the given object is an equivalent object, it’s essentially an alias of == (double-equals):

ruby
"Hello" === "Hello" # => true
"Hello" === "World" # => false

The === method on a Range will check if the given object is within the range:

ruby
(1..10) === 5 # => true
(1..10) === 15 # => false

Procs alias === to call:

ruby
is_even = proc { |it| it % 2 == 0 }
is_even === 42 # => true
is_even === 21 # => false

How Ruby uses types

Ruby uses these types in a few different ways already. For example, in a case statement, Ruby will use the === method of each case to determine if it matches the case.

ruby
case 42
when String
  puts "It's a string!"
when Integer
  puts "It's an integer!"
end

The above case statement is essentially equivalent to the following:

ruby
object = 42

if String === object
  puts "It's a string!"
elsif Integer === object
  puts "It's an integer!"
end

Ruby also uses types for pattern matching, both in and out of case statements.

ruby
case [42]
in Array[String]
  puts "It's a string in an array!"
in Array[Integer]
  puts "It's an integer in an array!"
end

The above pattern matching is essentially equivalent to the following:

ruby
object = [42]

if Array === object && String === object.deconstruct[0]
  puts "It's a string in an array!"
elsif Array === object && Integer === object.deconstruct[0]
  puts "It's an integer in an array!"
end

Ruby also uses types for methods on Arrays:

ruby
[1, 2, 3].all?(String) # false
[1, 2, 3].all?(Integer) # true

[1, "2", 3].any?(String) # true
[1, "2", 3].any?(Integer) # true

The methods all? and any? determine if all or any of the elements in the array match the given type by calling its === method. This is more efficient than passing a block and (depending on the type) usually avoids object allocations.

What about generics?

If a type is any object that responds to ===, then we can create generic functions that generate parameterized types.

Let’s make a generic function that generates a type that checks that an object is an array and all of its elements are of a certain type.

For this example, we can just make a method that generates a proc, since as we saw above, procs respond to ===. It would make sense to call this function Array, but since Ruby already has a method called Array, we’ll prefix it with an underscore _Array.

ruby
def _Array(type)
  proc { |it| Array === it && it.all?(type) }
end

Now we can use this generic function, passing it a type:

ruby
_Array(String) === [1, 2, 3] # false
_Array(Integer) === [1, 2, 3] # true

Literal has a number of built-in generics just like this.