The whole idea of this repos is to try and define jargon from combinatorics and category theory jargon that are used in functional programming in a easier fashion.
Let's try and define these with examples, this is a WIP—please feel free to send PR ;)
The number of arguments a function takes. From words like unary, binary, ternary, etc. This word has the distinction of being composed of two suffixes, "-ary" and "-ity." Addition, for example, takes two arguments, and so it is defined as a binary function or a function with an arity of two. Such a function may sometimes be called "dyadic" by people who prefer Greek roots to Latin. Likewise, a function that takes a variable number of arguments is called "variadic," whereas a binary function must be given two and only two arguments, currying and partial application notwithstanding (see below).
const sum = (a, b) => a + b;
const arity = sum.length;
console.log(arity);
// => 2
// The arity of sum is 2
A function which takes a function as an argument and/or returns a function.
const filter = (pred, xs) => {
const result = [];
for (var idx = 0; idx < xs.length; idx += 1) {
if (pred(xs[idx])) {
result.push(xs[idx]);
}
}
return result;
};
const is = type => x => Object(x) instanceof type;
filter(is(Number), [0, '1', 2, null]); //=> [0, 2]
The process of getting a function with lesser arity compared to the original function by fixing the number of arguments is known as partial application.
let sum = (a, b) => a + b;
// partially applying `a` to `40`
let partial = sum.bind(null, 40);
// Invoking it with `b`
partial(2); //=> 42
The process of converting a function with multiple arity into the same function with an arity of one. Not to be confused with partial application, which can produce a function with an arity greater than one.
let sum = (a,b) => a+b;
let curriedSum = (a) => (b) => a + b;
curriedSum(40)(2) // 42.
A function is said to be pure if the return value is only determined by its input values, without any side effects.
let greet = "yo";
greet.toUpperCase(); // YO;
greet // yo;
As opposed to:
let numbers = [1, 2, 3];
numbers.splice(0); // [1, 2, 3]
numbers // []
A function or expression is said to have a side effect if apart from returning a value, it modifies some state or has an observable interaction with external functions.
console.log("IO is a side effect!");
A function is said to be idempotent if it has no side-effects on multiple executions with the the same input parameters.
f(f(x)) = f(x)
Math.abs(Math.abs(10))
Writing functions where the definition does not explicitly define arguments. This style usually requires currying or other Higher-Order functions. A.K.A Tacit programming.
// Given
let map = fn => list => list.map(fn);
let add = (a, b) => a + b;
// Then
// Not points-free - `numbers` is an explicit parameter
let incrementAll = (numbers) => map(add(1))(numbers);
// Points-free - The array is an implicit parameter
let incrementAll2 = map(add(1));
total1
lists and uses the parameter numbers
, so it is not points-free. total2
is written just by combining functions and values, making no mention of its arguments. It is points-free.
It is easy to recognize points-free function definitions; they are the ones that contain no 'function
' keywords and no fat arrows.
A data structure that can have any other type, function, or data structure in it.
Array in JavaScript is a container type.
A container type that can be mapped over and returns the same kind of container.
Simplest functor in javascript is an Array
[2,3,4].map( n => n * 2 ); // [4,6,8]
Lift takes a function with n arguments and returns one that can be run on n containers of the same type.
Map is the same as a lift over a one-argument function:
lift(n => n * 2)([2,3,4]); // [4,6,8]
Unlike map lift can be used to combine values from multiple arrays:
lift((a, b) => a * b)([1, 2], [3]); // [3, 6]
Lift can apply to any Applicative Functor.
An expression that can be replaced with its value without changing the behavior of the program is said to be referentially transparent.
Say we have function greet:
let greet = () => "Hello World!";
Any invocation of greet()
can be replaced with Hello World!
hence greet is
referentially transparent.
When an application is composed of expressions and devoid of side effects, truths about the system can be derived from the parts.
Lazy evaluation is a call-by-need evaluation mechanism that delays the evaluation of an expression until its value is needed. In functional languages, this allows for structures like infinite lists, which would not normally be available in an imperative language where the sequencing of commands is significant.
let rand = function*() {
while(1<2) {
yield Math.random();
}
}
let randIter = rand();
randIter.next(); // Each exectuion gives a random value, expression is evluated on need.
A monoid is some data type and a two parameter function that "combines" two values of the type, where an identity value that does not affect the result of the function also exists.
The simplest monoid is numbers and addition:
1 + 1; // 2
The data type is number and the function is +
, the addition of two numbers.
1 + 0; // 1
The identity value is 0
- adding 0
to any number will not change it.
For something to be a monoid, it's also required that the order of operations will not affect the result:
1 + (2 + 3) == (1 + 2) + 3; // true
Array concatenation can also be said to be a monoid:
[1, 2].concat([3, 4]); // [1, 2, 3, 4]
The identity value is empty array []
[1, 2].concat([]); // [1, 2]
A monad is a container type that provides two functions, chain and ap. Monads provide an interface for executing a common sequence of commands on a particular kind of value, often one you want to avoid acting on directly. One of the most common monads is the "maybe" or optional value monad, which wraps a value that could be either nothing or something. By using a monad instead of the raw value, you can protect your code from exposure to null values. Likewise, a "state" monad can be used in a parser to algorithmically consume an input string using a repeatable sequence of steps that preserves the current state of the input from operation to operation. Also, since a monad is, by definition, a special kind of functor that also returns a monad, they can be chained together to describe any sequence of operations. In functional languages with lazy evaluation, monads are used where sequence of evaluation is important, such as in I/O. Due to this sequencing utility, they are sometimes referred to as "programmable semicolons."
The simplest monad is the Identity monad. It simply wraps a value.
let Identity = v => ({ chain: transform => transform(v) })
A chain is a container type that implements a chain function. The chain function takes another function to run on the contained value and returns a value in the same container. The passed function must also return a value in the same container. This is also known as bind, or flatmap in other languages.
['cat,dog','fish,bird'].chain((a) => a.split(',')) // ['cat','dog','fish','bird']
An applicative functor is a container type that can have functions put in it. A function often called
ap
is available on the type which applies a function in the container to a value in another container of the same type.
[(a)=> a + 1].ap([1]) // [2]
Two objects are Isomorphic is they satisfy the condition:
compose(to, from) == Identity
andcompose(from, to) == Identity
const toChars = [].join;
const fromChars = ''.split;
fromChars.call(toChars.call([1,2,3])) // [ '1,2,3' ]
toChars.call(fromChars.call([1,2,3])) // '1,2,3'