In this chapter we are going to play with another important property of functions - arity. In simple terms, arity represents the number of input parameters a function expects.

So, a function that does not take any parameters has an arity of zero, a function that takes one parameter has an arity of one, and so on.

In some programming languages, a function can have optional parameters, sometimes with a default value in case it was omitted in the function call.

Also, in some languages, a function may take a variable number of parameters. We call these variadic functions or varargs.

Let’s illustrate these points with some JavaScript examples:


//a function without any parameters has an arity of zero
let now = function(){
    return new Date();
}

//which we can inspect by looking up its length field
//note that we're not using () parentheses 
//that's because we're inspecting a field of the function object itself 
//and not the value it evaluates to
now.length
//=> 0

Of course, we know that function now is not pure. How would a pure zero-arity function look like?

let five = function(){
    return 5;
}
five.length
//=> 0

Since pure functions are only allowed to depend on its input parameters, and zero-arity pure functions have only one possible value of their inputs, which is nothing at all, we see that they are logically equivalent to constants, even if they are not implemented as such in a given programming language.

Let’s see some more examples of functions with different arities:

let square = function(x){
    return x * x;
}
square(5)
//=> 25
square.length
//=> 1

let subtract = function(a, b){
    return a - b;
}
subtract(10, 2)
//=> 8
subtract.length 
//=> 2

let greet = function(name){
    return "Hello " + name;
}
greet("Fred")
//=> "Hello Fred"
greet.length
//=> 1

let sum = function(a, b, c){
    return a + b + c;
}
sum(1, 2, 3)
//=> 6
sum.length
//=> 3

Now is probably a good time to mention that JavaScript has a very relaxed attitude to the number of function parameters. You can actually pass as many as you want, it won’t complain! Whether it will do anything sensible with them is a different matter, though.

If you pass fewer parameters than the function expects, the ones you omitted will be treated as undefined.

If you pass more parameters than the function expects, the extra ones will be simply ignored.

We can use this to implement a function with optional parameters. Let’s say that we want to have the name parameter in our greet function optional. If it is omitted, the function should say “Hello human”.

let greet = function(name){
    if(name === undefined){
        name = "human"
    }
    return "Hello " + name;
}

//if we pass the name, it works as before
greet("Fred")
//=> "Hello Fred"

//if we omit it, it uses the default override
greet()
//=> "Hello human"

//and if we add a bunch of extra parameters, they are all ignored
greet("Fred", "Barney", "Wilma")
//=> "Hello Fred"

greet.length
//=> 1

In this example, we never explicitly marked the parameter as optional in the function signature. Instead, we defined the default behavior in the function body, so JavaScript interpreter had no way to infer that the parameter is actually optional and decided that the arity is one.

A nicer way to achieve the same effect is to write a function and assign a default value to the parameter:

let greet = function(name = "human"){
  return "Hello " + name;
}

greet("Fred")
//=> "Hello Fred"

greet()
//=> "Hello human"

greet.length
//=> 0

In this example, we explicitly set the parameter as optional by supplying the default value and, as it turns out, JavaScript does not count optional parameters towards the function arity.

Note that, while JavaScript considers the first example to have an arity of one and the second one an arity of zero, this is a language implementation decision. Theoretically speaking, we can say that both function implementations have an arity of zero or one, i.e. it is a variable arity function with two possible arities.

But what about proper variadic functions, the ones that can take any number of parameters? Let’s take another look at our sum function from our previous examples.

let sum = function(a, b, c){
    return a + b + c;
}

sum()
//=> NaN

sum(1)
//=> NaN

sum(1, 2)
//=> NaN

sum(1, 2, 3)
//=> 6

sum(1, 2, 3, 4)
//=> 6

We can only use it to calculate a sum of exactly three numbers. That’s not very useful! Ideally, we should be able to calculate the sum of any number of parameters.

The first workaround that comes to our mind is to wrap all the numbers in a single array parameter:

let sum = function(numbers){
    let result = 0
    for(let i = 0; i < numbers.length; i++){
        result += numbers[i]
    }
    return result;
}

sum([])
//=> 0

sum([1])
//=> 1

sum([1, 2])
//=> 3

sum([1, 2, 3])
//=> 6

sum([1, 2, 3, 4])
//=> 10

This technically works, but having to wrap the numbers with square brackets is just not very nice. Can we do any better?

let sum = function(){
    let result = 0
    for(let i = 0; i < arguments.length; i++){
        result += arguments[i]
    }
    return result;
}

sum()
//=> 0

sum(1)
//=> 1

sum(1, 2)
//=> 3

sum(1, 2, 3)
//=> 6

sum(1, 2, 3, 4)
//=> 10

JavaScript functions have a hidden parameter called arguments which is an array-like object with all the parameter values which were actually sent to it during the function call, regardless of how many parameters the function declared. It’s a bit ugly, but it works.

Luckily, as of ES6, we have native support for varargs in JavaScript, and we can use a bit nicer syntax:

let sum = function(...numbers){
    let result = 0;
    for(let i = 0; i < numbers.length; i++){
        result += numbers[i]
    }
    return result;
}

It is all nice and good to have a function that accepts a lot of different parameters. It lets us parameterize just about anything. If we need to do so, that is. Sometimes, however, we need to do the opposite, i.e. to reduce the number of function parameters.

Let’s say that we have a somewhat convoluted function that says goodbye (to complement our existing greet function):

let goodbye = function(greeting, mood, name){
    let moodSignal = "";
    if(mood === "excited"){
        moodSignal = "!! :)"
    }else if(mood === "sad"){
        moodSignal = "... :("
    }
    return greeting + ", " + name + " " + moodSignal;
}

goodbye("Goodbye", "sad", "Marta")
//=> "Goodbye, Marta... :("

goodbye("Arivederci", "neutral", "Vito")
//=> "Arivederci, Vito"

goodbye("See you later", "excited", "human")
//=> "See you later, human!! :)"

Having many options to configure is good, but what if we want to reuse the same greeting type and mood for many function calls where only the recipient name is different? We can use a partial function for that.

Simply put, a partial function is a function where some parameter values are fixed, typically left to right, while others remain free to be passed in function calls.

We can create a partial function from a regular function using this fairly simple higher order function:

function partial(f, ...fixedParams){
  return function(){
    return f(...fixedParams, ...arguments)
  }
}

let farewell = partial(goodbye, "Farewell", "neutral");

farewell("Ernest")
//=> "Farewell, Ernest"

farewell("Susan")
//=> "Farewell, Susan"

Our function partial takes the original function and a number of parameters to fix, and it returns another function with these parameters prepended to any other parameters passed to it.

And we can even create partial functions from original variadic functions, e.g:

let sumWithHeadStart = partial(sum, 10, 20, 30);
sumWithHeadStart(1)
//=> 61

While they embody a very simple and elegant concept, partial functions unfortunately have limited usability. Specifically, in order for them to be practical, we must have an original function where the parameters we want to fix are in the front and the ones we want to keep variable in the back of the parameter list.

Imagine if we had a different scenario and wanted to always say goodbye to the same person, in the same mood, but using different greeting phrases. In that case, we would have simply had to write a custom function:

let goodbyeFred = function(greeting){
    return goodbye(greeting, "neutral", "Fred")
}

goodbyeFred("See ya")
//=> "See ya, Fred"

Partial functions are often confused with a similar concept called currying.

Curried functions are such functions that depend on more than one parameter, but rather than to have one function accepting all parameters and returning the value, they consist of a sequence of functions where each takes exactly one parameter and returns another curried function with the remaining parameters.

In our example, a curried version of goodbye:

  1. would take only greeting as the parameter and return a function which
  2. takes only mood as its parameter and returns another function which
  3. takes name as the parameter and returns the final value.
//we call a regular function like this
goodbye("Arivederci", "neutral", "Vito")

//and a curried one like this
goodbyeCurried("Arivederci")("neutral")("Vito")

Let’s try out implementing goodbyeCurried:

let goodbyeCurried = function(greeting){
    return function(mood){
        return function(name){
            return goodbye(greeting, mood, name);
        }
    }
}

goodbyeCurried("Arivederci")("neutral")("Vito")
//=> "Arivederci, Vito"

And that works! We can also generalize the concept and write a high order function that accepts a regular function and returns its curried equivalent:

function curry(f, arity = f.length){
  if(arity < 2){
    return f;
  }
  return function(a){
    return curry(function(){
      return f(a, ...arguments);
    }, arity - 1);
  }
}

//now we can get a curried version of any function
//(as long as it has an exact number of parameters!)
let goodbyeCurried = curry(goodbye)
goodbyeCurried("See you later")("excited")("human")
//=> "See you later, human!! :)"

Note that we’re inspecting the original function’s arity. That’s necessary, since we need to have an exit clause for our recursion. In other words, we can’t keep returning other functions forever, we have to return the actual value at some point and keeping track of arity makes that possible.

The consequence is that we can’t curry variadic functions, but that actually makes sense as such a function would keep returning other functions. It wouldn’t know when to stop and just return the value.

To summarize, the main differences between partial and curried functions are:

  • Partial functions may take more than one parameter, while curried functions only ever take one
  • Partial functions immediately return the value, while curried functions only return the next function (unless we are calling the very last function in the sequence)

We can conclude this chapter with a (hopefully) fun fact. Curried functions owe their name to a mathematician called Haskell Curry whose work became the theoretical foundation for the concepts we were just exploring. Such was his impact and legacy in this field that currying is not the only thing named after him, but also a programming language Haskell, which, perhaps unsurprisingly, uses currying as its default way of handling function parameters.

-- function add takes only one parameter - x
-- and returns another function that takes only one parameter - y
-- and returns the final value.
-- Parentheses added for emphasis, they are otherwise optional
add :: Integer -> (Integer -> Integer)   
add x y =  x + y