ES6 workshop
Contents
- let and const
- Template literals
- Enhanced object literals
- Object destructuring
- Rest operator
- Spread syntax
- Arrow functions
- Symbols
- Set and Map
- Iterators
- Generators
- Proxies
Short history
-
1995
- May - Mocha invented by Brendan Eich at Netscape (in 10 days!)
- September - renamed to LiveScript
- December - renamed to JavaScript (because JAVA...)
- 1996 - ECMA takes JavaScript to standardization
- 1997 - ECMA-262 (ECMAScript)
- 1998 - ECMAScript 2
- 1999 - ECMAScript 3
- 2005 - Mozilla and Macromedia began to work on ECMAScript 4 - never released => ECMAScript 3.1
- 2009 - ES5 emerges from previous discussions
-
2015 - ES6 standard
- Version names based on year of release (ES2015, ES2016
let and const
var is function scoped, cons and let are block (lexical) scoped
- var declarations are hoisted to the top of the function
- let declarations hoisted to the top of the block but cannot be used before initialization (temporal dead zone)
- const declarations must be initialized
- let and const cannot be redeclared
- let allows for reassignment, const does not (not the same as immutable - for immutability we can use Object.freeze and/or Object.seal)
Template literals
- string literals that allow embedded expressions
- back-ticks (`) are used to enclose the string
- anything inside ${} is evaluated
- new lines are retained
- tag templates - customize the resulting output string - useful for l10n/i18n or html sanitising
Enhanced object literals
-
property initializer shorthand
-
concise method names
-
computed property names
-
new Object methods
-
Object.is() - make up for the remaining quirks of the identically equals operator
-
Object.assign(receiver, suppliers) - copy properties from supplier into receiver
-
Object.getOwnPropertyNames() - return all properties of an object (as opposed to only enumerable for Object.keys())
-
Object destructuring
-
de-structure a target objects' properties
-
works on all objects (including arrays)
-
can have default values for undefined properties
-
default values evaluation is similar to a list of let variables declaration
Rest operator
-
rest parameters for functions and rest items for arrays
-
use the ... operator to:
-
represent an indefinite number of function parameters as an array
-
to assign remaining values in an array to a particular value
-
Spread syntax
allows an expression to be expanded in places where multiple arguments (for function calls) or multiple elements (for array literals) or multiple variables (for destructuring assignment) are expected
a replacement for apply
easier array operations
easier copy operation for arrays and objects
Arrow functions
-
shorthand for the function expression
-
does not bind its own this, arguments, super or new.target
-
preserves the 'this' of the calling location
-
best suited for non-method functions
-
are always anonymous
-
cannot be used as constructors and do not have a prototype
-
does not bind to an arguments object -> use rest params
Recap
function() {
if(true) {
var x = 1;
}
console.log(x) // 1
}
Block level scoping: const and let
function() {
for(var i=0;i< 10;i++) {
var x = 1;
}
console.log(i) // 10
console.log(x) // 1
}
function() {
if(true) {
let x = 1;
}
console.log(x) // x doesn't exist here
}
function() {
let x = 1;
if(true) {
let x = 2; // error cannot reasign
}
}
function() {
const a; // error, need to initialize const
const b = 1;
b = 2; // error cannot reasign
}
function() {
for(var i=0; i<10; i++) {
let x = 1;
}
console.log(i, x) // i, x don't exist here
}
Recap
var firstName = "Andrei",
lastName = "Antal";
var fullName = "The full name is: " + firstName + " " + lastName
Template strings
var fullName = `The full name is: ${firstName} ${lastName}`;
function() {
return (new Date()).year;
}
var currentYear = `The cuurent Year is ${getYear()}`;
var theSum = `The sum is ${3 + 4 / 2}`;
const theString = upp`Hello ${firstName} ${lastName}!`;
function upp(literal, name, name2) {
return [
literal[0],
name,
literal[1],
name2.toUpperCase(),
literal[2],
].join("");
}
Recap
const name = "Andrei";
const age = 29;
const user1 = {
name: name,
age: age,
sayHello: function() {
console.log("Yo!");
}
}
Enhanced object literals
const user2 = {
name,
age,
sayHello() {
console.log("Yo!");
}
}
const firstName = "first name";
const suffix = " name";
const person = {
[firstName]: "Andrei",
["last" + suffix]: "Antal"
};
const obj1 = {
a: 'a',
b: 'b',
};
const obj2 = { b: 'c' };
Object.assign(obj1, obj2); // obj1 = { a: 'a', b: 'c' }
Object.assign({}, obj1, obj2); // { a: 'a', b: 'c' }
Recap
let options = {
repeat: true,
save: false
};
let repeat = options.repeat;
let save = options.save;
Destructuring
let {repeat, save} = options;
console.log(repeat); // true
console.log(save); // false
let {repeat: re, save: sa, stop: st = false} = options;
console.log(re, sa, st); // true, false, false
let [one, two, three] = [1,2,3];
console.log(one, two, three); // 1,2,3
let [, , three] = [1,2,3];
console.log(three); // 3
Recap
const animal = {
name: 'Dog',
sound: 'wof'
};
function makeSound(options) {
options.name = options.name || 'animal';
console.log(`The ${options.animal} goes ${options.sound}`)
}
Destructuring
function makeSound({name = 'animal', sound}) {
console.log(`The ${name} goes ${sound}`)
}
makeSound(animal);
Recap
function doSomething(a,b, ...rest) {
console.log(a);
console.log(b);
console.log(rest);
}
doSomething(1,2,3,4,5); // 1, 2, [3,4,5]
function doSomething(a,b, ...rest, c){} // error; rest must be the last in param list
Rest operator
const arr = [1, 2, 3, 4, 5];
const [,, ...arr1] = arr;
console.log(arr1); // [3, 4, 5]
function doSomething(a,b, ...[c,d,e]){
console.log(a);
console.log(b);
console.log(c,d,e);
}
doSomething(1,2,3,4,5);
Recap
function logNumbers(a, b, c) {
console.log(a)
console.log(b)
console.log(c)
}
const array = [1,2,3];
logNumbers(...array);
Spread
const array = [1,2,3];
console.log([...array, 4, 5]);
const obj1 = {a: 'a', b: 'b'};
const obj2 = {c: 'c', ...obj1};
console.log(obj2);
const arr1 = [1,2,3];
const arr2 = [2,3,4];
const arr3 = [...arr1, 8, ...arr2];
console.log(arr3);
Recap
const sum = (num1, num2) => { return num1 + num2 };
const sum = function(num1, num2) {
return num1 + num2;
};
Arrow functions
const square = value => value * value;
const square = function(value) {
return value * value;
};
const getName = () => "Andrei";
const getName = function() {
return "Andrei";
};
const getTempItem = (id) => ({ id: id, name: "Temp" });
const getTempItem = function(id) {
return {
id: id,
name: "Temp"
};
};
Recap
const sum = (num1, num2) => { return num1 + num2 };
const sum = function(num1, num2) {
return num1 + num2;
};
Arrow functions
const square = value => value * value;
const square = function(value) {
return value * value;
};
const getName = () => "Andrei";
const getName = function() {
return "Andrei";
};
const getTempItem = (id) => ({ id: id, name: "Temp" });
const getTempItem = function(id) {
return {
id: id,
name: "Temp"
};
};
Classes
-
syntactic sugar over over JavaScript's existing prototype-based inheritance - does not create a new object oriented inheritance model
-
classes are defined using the class keyword
-
class declarations are not hoisted
- a class contains
- constructor - class initialization
- methods (we can also create also static methods)
- properties (only using getters)
- we can create hierarchies using the extends keyword
- we ca use super() to call an object's parent
Symbols
-
a primitive (value) data type whose instances are unique and immutable
-
for debugging purposes symbols can have an optional description - stored in the [[Description]] property
-
symbols do not have a literal form, so they do not coerce into other primitive types
-
mostly used to create private properties - symbol keys are not visible in iterations
-
ES6 provides a global symbol registry in order to share symbols across the app
Set and Map
-
extra data types for working with collections (besides the Array data type)
-
can store primitive or object values
-
Set - iterable list of objects that cannot contain duplicates; elements are set in the insertion order
-
Map - iterable key/value pair entries list - keys can be objects, symbols, primitives and functions
-
WeakMap, WeakSet - non enumerable versions where keys/entries are referenced weakly - can be garbage collected if they aren't referenced elswhere
Iterators
- iterators - objects with special interfaces designed for iteration
- affects behaviour in for...of loops and the ... operator
- the iterable protocol - customize iteration behaviour
- implement @@iterator method (or [Symbol.iterator]) that returns an iterator
- the iterator protocol - implement a next() method that returns an object with 2 keys:
- value - current value in iteration
- done - boolean (true when there are no more items in the enumeration)
Generators
-
generator - object is returned by a generator function and it conforms to both the iterable protocol and the iterator protocol.
-
declared by adding a "*" between function and the name
-
use the yield keyword to denote execution halt until the next() method is called
-
each yield return an iteration (value, done) object
Proxies
-
proxy objects are used to define custom behavior for fundamental operations
-
create a proxy to use in place of another object (called the target)
-
intercept low-level object operations on the target (using a trap) that are otherwise internal to the JavaScript engine
-
use the Proxy constructor to make a proxy - pass it two arguments: the target and a handler (an object that defines one or more traps)
Other ES stuff
-
module system
-
native promises
-
tail call optimization
ES6 workshop
By Andrei Antal
ES6 workshop
- 1,248