All values have a type. The data type determines what you can do with a given value.
There are five main primitive data types: Number, String, Boolean, undefined, and null
// Strings must have matching "" or '' or ``
const dQuoteString = "Double quotes are nice when a string has a ' in it";
const sQuoteString = 'Single quotes are nice for the "opposite" reason.';
const template = `Template strings allow us to embed expressions like ${5 > 3 ? "yay" : "boo"}`;
// All of these are the Number data type
const number = 10;
const float = 1.5;
const integer = 1;
const negative = -1;
// NaN means Not a Number, but it's technically a Number type.
// They are created when you try to do math on non-Number types.
const notANumber = NaN;
const alsoNaN = 5 * "hello";
// Booleans only have two values. They are case-sensitive!
let boolean = true;
boolean = false;
// Undefined is a type with only one value: undefined
const explicitUndefined = undefined;
const undefinedVariable; // Assigned undefined by default
// Null is a type with only one value: null
// null must be assigned to a variable explicitly
const nullVariable = null;
There are also three reference data types: Objects, Arrays, and Functions (we'll talk about these later on).
Technically, there is only one reference data type: Objects. Functions and Arrays are subsets of Objects
Variables
A Variable in JavaScript is a named container for data.
There are 4 ways to declare a new variable, but only the first two of should be used:
const declares a block-scoped variable that cannot be reassigned. This should be your go-to variable declaration.
let declares a re-assignable block-scoped variable. Use only when reassignment is necessary.
var declares a re-assignable, hoisted, function-scoped variable. Do not use var. Hoisting and function-scoped variables can cause unexpected behavior (read more below).
Global variables are declared without a keyword. Do not use global variables.
// Good
const myName = 'John';
// Uncaught TypeError: Assignment to constant variable. Hmm, time to use a let.
myName = 'Johnny';
// Also good
let myAge = 101;
// No error!
myAge = 5;
// Bad
var status = 'happy';
// Very bad
globalStatus = 'sad';
We can reference a variable by name to either reassign its value or to access its current value:
// use let because we will reassign this variable
let count = 0;
// reference the count variable to access its current value, 0
let message = `the count is ${count}`;
// Prints "the count is 0"
console.log(message);
// reference the count variable to reassign it
count = 1;
// reference the count variable again to access its new value, 1
message = `the count is now ${count}`
// Prints "the count is now 1
console.log(message);
Hoisting: Why We Don't Use var
TLDR: Don't use var or function. Do not reference a variable or function before it has been declared.
Hoisting is a feature of JavaScript that allows certain variables/functions to be referenced before they are declared.
Variables declared with the var keyword and Functions declared with the function keyword are hoisted.
Variables declared with const or let and Arrow Functions are NOT hoisted.
Variables declared with var will be given the value undefined until their assignment statement is reached.
Functions declared with function are fully invoke-able anywhere within their scope.
// Good — Uncaught ReferenceError: Cannot access 'name' before initialization
console.log(`Hi ${name}!`);
const name = 'Jane';
// Also Good — Uncaught ReferenceError: Cannot access 'age' before initialization
console.log(`I'm ${age}.`);
let age = 101;
// Bad! — Prints "Hi, undefined?" with no Error
console.log(`Hi, ${weird}?`);
var weird = 'Tom';
// Confusing! — Throws an error but only because `undefined` doesn't have a `toUpperCase` method
console.log(str.toUpperCase());
var str = 'Some string';
// Bad! - This works but it really shouldn't
foo();
function foo() {
console.log('foo!');
}
Hoisting (like most weird JS things) stems from the fact that when JS was invented eons ago, there was a desire that JS should never crash a site. Ever. So bugs that would normally get caught in development would sneak through so the page wouldn’t crash, but it also wouldn’t work properly. Nowadays, we prefer to find these errors in development, so crashing is ok.
Functions
A Function is a named container for statements that can be invoked to execute its statements.
Creating Arrow Functions
There are many ways to create a function, but we will use Arrow Function syntax:
// Best — Arrow Functions are stored in constant variables and aren't hoisted.
const multiply = (x, y) => {
return x * y;
}
// Not Great — Function Declaration. It is hoisted. Don't do this.
function add(x, y) {
return x + y;
}
Function Calls
After a function is declared, the function must be invoked to use the function's code using the syntax:
// adding parentheses after the function name invokes the function
functionName();
Invoking (or "calling") a function causes the program to jump to the first line of code in the function, setting it as the next line of code to be executed.
console.log('This happens first');
const sayHello = () => {
// not executed until sayHello() is invoked
console.log('hello');
}
console.log('this happens second');
// hello is printed when sayHello() is invoked
sayHello();
// we can invoke a function multiple times!
sayHello();
Note that the order in which statements are executed in our code is not always top to bottom. Declaring the function doesn't cause the code inside to run. We only execute the code inside of sayHello when it is invoked a few lines later.
Parameters and Arguments
The function sayHello above has no flexibility in its functionality. It always prints "hello".
Parameters are variables created inside of a function that enable a function to change its behavior based on arguments provided in the function call.
Parameters go in the parentheses of a function declaration (separated by commas).
Arguments go in the parentheses of the function call (also separated by commas).
The order of the arguments must match the order of the parentheses.
// printSum can add and print any two given values
// x and y are parameters that reference the values provided when the function is called.
const printSum = (x, y) => {
console.log(x + y);
}
// 5 and 3 are the arguments. 8 is printed
printSum(5, 3);
// This time, 10 and 2 are the arguments. 12 is printed
printSum(10, 2);
// There is no type-checking, the function will just concatenate these values
// 'hello5' is printed
printSum('hello', 5);
// If an argument is not provided, the parameter will be undefined
// undefined + undefined => NaN is printed
printSum();
Return Statements
The functions above use console.log to print out a result to the console, but that result can't be used later in the program. If we want a function to produce a value that we can be used outside of the function, we add a return statement.
A return statement does two things:
It terminates the execution of the function
It returns a value to the location of the function call.
const add = (x, y) => {
return x + y;
}
// The function call is replaced by its return value, so:
// const sum = 8
const sum = add(5, 3);
console.log(sum); // Prints 8
We say that "add(5, 3) resolves to 8"
A function can have multiple return statements, but only one is ever executed. This is helpful if the value returned depends on the provided argument.
If the function's body is just a return statement, you can omit the {} and the return keyword. This is called an implicit return
// Totally Fine - An explicit return
const add = (a, b) => {
return a + b;
}
// Better - An "implicit return"
const add = (a, b) => a + b;
Scope
Scope refers to where in our code variables can be referenced. Files, functions, and code blocks {} each create a new scope.
There can be any number of scopes in a single file but we will describe them as either:
Global Scope — Variables that are accessible across the entire file. Includes things like console and Math and variables/functions declared at the outermost scope of the file.
Local Scope — Variables that are accessible only in the current function or code block.
"Outer scopes" cannot see variables in "inner scopes".
index.js
// Global Scope - secretValue is accessible anywhere in the file.
const secretValue = () => {
// Local Scope - this variable is only accessible in this code block
const private = 'shhhhh!';
// Return statements let us send values out of an "inner" scope, but the function must be invoked to do this.
return private;
}
// We can't access private because we are in an "outer" scope compared to where it is declared.
console.log(private); // ReferenceError: private is not defined
// We can get the value from the function if we invoke it though
console.log(secreteValue()); // Prints "shhhhh!"
"Inner scopes" can see variables in "outer scopes".
Parameters are treated as locally scoped variables.
index.js
// Global Scope - friend is accessible anywhere in this file.
const friend = 'John';
// Global Scope - greetFriend is also accessible anywhere in the file.
const greetFriend = (myName) => {
// The myName parameter has local/function scope. It can only be accessed inside of greetFriend
// We can access `friend` from greetFriend since we are in an "inner" scope.
console.log(`Hi, ${friend}, I'm ${myName}`);
}
greetFriend('Jane'); // Prints "Hi, John, I'm Jane"
// We can't access myName because we are in an "outer" scope compared to it is declared.
myName; // ReferenceError: myName is not defined
Zooming in on Strings
String Indexes and Bracket Notation
A string is a series of characters surrounded by "double quotes", 'single quotes' or `backticks`
Each character in a string, including spaces, has an index — a numbered position starting at 0.
Bracket Notation is used to read a single character in a string by its index: string[index]
const message = 'Hello there!';
console.log( message[0] ); // Prints 'H'
console.log( message[4] ); // Prints 'o'
console.log( message[5] ); // Prints ' '
console.log( "abc"[1] ); // Prints 'b'
If you use bracket notation and there is no character at the given index, undefined will be returned:
Every string also has a length property that is accessed using dot notation: string.length
const message = 'Hello there!';
console.log( message.length ); // Prints 12
// The length property is useful for calculating indexes from the end of the string
console.log( message[message.length - 1] ); // Prints "!"
console.log( message[message.length - 2] ); // Prints "e"
console.log( message[message.length - 12] ); // Prints "H"
const firstAndLastAreTheSame = (str) => {
return str[0] === str[str.length-1]
}
console.log(firstAndLastAreTheSame('hello')); // Prints false
console.log(firstAndLastAreTheSame('helloh')); // Prints true
Strings are Immutable
Strings are "immutable" (unable to be mutated). In other words, you cannot use bracket notation to change characters of a string.
If you try, no error will be thrown, nothing will happen.
const message = 'Hello there!';
// Nothing will happen if you attempt to mutate a string.
message[0] = "J";
console.log(message); // Prints "Hello there!"
String Methods
A method is a function that is attached to a value. Often, methods are used to manipulate the value they are attached to.
Methods are invoked using dot notation: value.method()
First, let's look at some "read only" string methods: includes, startsWith, endsWith, indexOf, lastIndexOf
const fruits = 'apples, bananas, cherries';
// These return a boolean
console.log( fruits.includes('ana') ) // Prints true
console.log( fruits.includes('ANA') ) // Prints false
console.log( fruits.startsWith('apple') ) // Prints true
console.log( fruits.endsWith('s') ) // Prints true
// These return an index number (or -1 if not found)
console.log( fruits.indexOf('p') ) // Prints 1
console.log( fruits.lastIndexOf('p') ) // Prints 2
console.log( fruits.indexOf('?') ) // Prints -1
const hasOnlyOne = (str, letter) => {
return str.indexOf(letter) === str.lastIndexOf(letter);
}
console.log(hasOnlyOne('hello', 'h')) // Prints true
console.log(hasOnlyOne('hello', 'l')) // Prints false
The following methods make a copy of a string before performing some modifications: slice, toUpperCase, toLowerCase, repeat, replace, replaceAll
const fruits = 'apples, bananas, cherries';
const apples = fruits.slice(0, 6);
console.log(apples); // Prints "apples"
const loudFruits = fruits.toUpperCase()
console.log(loudFruits); // Prints 'APPLES, BANANAS, CHERRIES'
console.log(apples.repeat(3)); // Prints 'applesapplesapples'
console.log(apples.replace('p', 'g')); // Prints 'agples'
console.log(apples.replaceAll('p', 'g')); // Prints 'aggles'