Dorokhov.codes

02. Functions

Functions

Let’s start with the easiest conception - a function.

Negotiation: a function name should be a verb.

Rest operator:

function customFunciton(a, b, ...rest) {
    // ... rest is an array of additional parameters.
}

Function declaration

function getRectArea(width, height) {
  return width * height;
};

We can use such functions even before declaration.

Function expression

var getRectArea = function(width, height) {
  return width * height;
};

We can use such function only after initiating a variable.

Arrow functions

In ES6 and later, there is a shorthand syntax for defining functions. This concise syntax uses => to separate the argument list from the function body, so functions defined this way are known as arrow functions.

Arrow functions are most commonly used when you want to pass an unnamed function as an argument to another function. The preceding code looks like this when rewritten to use arrow functions:

Arrow function doesn’t have its this context, but it takes this context from a parent function.

const plus1 = x => x + 1;        // The input x maps to the output x + 1
const square = x => x * x;      // The input x maps to the output x * x

plus1(y)                        // => 4: function invocation is the same
square(plus1(y))                // => 16
// Equal definitions:
const calc = (a, b) => a + b;
const calc = (a, b) => { return a + b; };

Callback functions

Docs.

function showName(name, callback) {
    console.log(name);
    callback();
}

showName("Andrew", () => {
    console.log("Done!");
});

Closures

Detailed description of closures.

Before understanding closures let’s understand what is lexical environment. In JavaScript, every running function and code block {...} have an internal (hidden) associated object known as the lexical environment.

The Lexical Environment object consists of two parts:

  • Environment Record – an object that stores all local variables as its properties (including this).
  • A reference to the outer lexical environment, the one associated with the outer code.

Roughly speaking, a running function has links to the inner variables (inner lexical environment), and has links to the outer variables (outer lexical environment). Moreover, these environment links are created at the time the function is created, and not when or where it is called.

function makeCounter() {
  let count = 0;

  return function() {
    return count++;
  };
}

let counter = makeCounter();

In this example, counter will have access to count because it is still placed in its outer lexical environment.

// Lexical environment of the function refers to the outer lexical environment object
// which still has the "count" variable:
// counter.[[Environment]] -> {count: 0} 

Use Scope section in your JS debugger to study this.

After a function is finished executing, the inner lexical environment is destroyed.

When a function is called, a new inner lexical environment is created.

A closure is a function that remembers its outer variables and can access them. In JavaScript, all functions are naturally closures.

IIFE

Immediately invoked function expression:

(function () {
    // ...
}());
(function () {
    // ...
})();

Example of how we can create an object interface:

const user = (function () {
    const getName = function() {};
    return {
        getName: getName 
    };
}());

Context “this”

this (aka “the context”) is a special keyword inside each function and its value only depends on how the function was called, not how/when/where it was defined.

this always refers to some object.

Example:

var addColorField = function() {
    this.color = "red";
}

var someObject = {
    height: 150,
    addColor: addColorField
};

someObject.addColor();

// someObject => {height: 150, color: 'red', addColor: ƒ}

If we call addColorField() in a browser, we will set Window.color property.

function test() {
    console.log(this); // we will get the `window` object.
}
'use strict';
function test() {
    console.log(this); // we will get 'undefined'.
}

Inject object context into a function:

function test() {
    console.log(this.name);
}

const user = {
    name: "Andrew"
};

test.call(user); // If a function has parameters, we will pass them using commas.
test.apply(user); // If a function has parameters, we will pass them in an array.

bind

Every function has the method .bind, which returns a new function with this bound to a value. The function has exactly the same behavior as the one you called .bind on, only that this was set by you. No matter how or when that function is called, this will always refer to the passed value.

var f = setInterval(function() {
    // ...
}.bind(this));

Creating a new function using bind:

function count(num) {
    return this * num;
}

const double = count.bind(2); // We will get a permanent function where num is always 2.