JavaScript The Hard Parts, v2
Introduction
Overview
- JavaScript principles
- Callbacks & High Order functions
- Closure
- Classes/Prototypes & Asynchronicity
What to focus on in the workshop?
- Analytical problem solving
- Engineering approach
- Technical communication
- Non-technical communication
- JavaScript and programming experience
Junior engineer takes any feature they’re given to build. And if they’ve seen technology or maybe the solution before, they can solve it.
Mid-level engineer takes any feature they’re given to build. And even if they’ve not seen the technology or solution before, they can figure it out because they’ve learned how to learn, they’re problem solving strong.
Senior engineer is somebody who can take any given feature, and not only just figure for themselves, but enable the rest of their team to figure it out. Because they have technical communication, the ability to explain what their code is doing to somebody else in their team, in a clear and cordial manner.
JavaScript Principles
Thread of Execution
- When JavaScript code runs, it:
- Goes through the code line-by-line and runs/‘executes’ each line - known as thread of execution
- Saves ‘data’ like strings and arrays so we can use that data later - in its memory
- We can even save code(‘functions’)
1 | const num = 3 |
Functions
Code we save (‘define’) functions & can use (call/invoke/execute/run) later with the function’s name & ()
Execution context
- Created to run the code of a function - has 2 parts(we’ve already seen them)
- Thread of execution
- Memory
- Created to run the code of a function - has 2 parts(we’ve already seen them)
Call Stack
- JavaScript keeps track of what function is currently running(where’s the thread of execution)
- Run a function - add to call stack
- Finish running the function - JS removes it from call stack
- Whatever is top of the call stack - that’s the function we’re currently running
Functions & Callbacks
Generalized Functions
- Why do we even have functions?
- Let’s see why…
- Create a function 10 squared
- Takes no input
- Returns 10 * 10
- What is the syntax(the exact code we type)?
1 | function tenSquared() { |
What about a 9 squared function?
What about a 8 squared function? 125 squared function?
What principle are we breaking?
DRY
We can generalize the function to make it reuseable
1 | function squareNum(num) { |
- Generalizing functions
- ‘Parameters’(placeholders) mean we don’t need to decide what data to run our functionality on until we run the function
- Then provide an actual value(‘argument’) when we run the function
- High order functions follow this same principle
- We may not want to decide exactly what some of our functionality is until we run our function
- ‘Parameters’(placeholders) mean we don’t need to decide what data to run our functionality on until we run the function
Repeating Functionality
- Now suppose we have a function
copyArrayAndMultiplyBy2
1 | function copyArrayAndMultiplyBy2(array) { |
- What if we want to copy array and divide by 2?
- Or add 3?
High Order Functions
- We could generalize our function - So we pass in our specific instruction only when we run copyArrayAndManipulate!
Higher Order Functions Example
1 | function copyArrayAndManipulate(array, instructions) { |
Callbacks & Higher Order Functions
How was this possible?
- Functions in JavaScript = first class objects
- They can co-exist with and can be treated like any other JavaScript object
- Assigned to variables and properties of other objects
- Passed as arguments into functions
- Returned as values from functions
Which is our Higher Order Function?
- The outer function that takes in a function is our higher-order function
Which is our Callback Function?
- The function we insert is our callback function
Higher-order functions
- Takes in a function or passes out a function
- Just a term to describe these functions - any function that does it we call that - but there’s nothing different about them inherently
Callbacks and Higher Order Functions simplify our code and keep it DRY
- Declarative readable code: Map, filter, reduce - the most readable way to write code to work with data
- Codesmith & pro interview prep: One of the most popular topics to test in interview both for Codesmith and mid/senior level job interviews
- Asynchronous JavaScript: Callbacks are a core aspect of async JavaScript, and are under-the-hood of promises, async/await
Arrow Functions
- Introducing arrow functions - a shorthand way to save functions
1 | function multiplyBy2(input) { return input * 2 } |
- Updating our callback function as an arrow function
1 | function copyArrayAndManipulate(array, instructions) { |
- We can even pass in multiplyBy2 directly without a name
- But it’s still just the code of a function being passed into copyAndManipulate
1 | function copyArrayAndManipulate(array, instructions) { |
- Anonymous and arrow functions
- Improve immediate legibility of the code
- But at least for our purposes here they are ‘syntactic sugar’ - we’ll see their full effects later
- Understanding how they’re working under-the-hood is vital to avoid confusion
Pair Programming
Pair Programming
- The most effective way to grow as a software engineer
Researcher
- Avoids blocks by reading everything they can find on their block/bug
Stackoverflower
- Uses code snippets to fix bug without knowing how they work
Pair programming
- Tackle blocks with a partner
- Stay focused on the problem
- Refine technical communication
- Collaborate to solve problem
For each topic you know give yourself a point to get a total out of
- I know what a variable is
- I’ve created a function before
- I’ve added a CSS style before
- I have implemented a sort algorithm(bubble, merge, etc)
- I can add a method to an object’s prototype
- I understand the event loop in JavaScript
- I understand ‘callback functions’
- I can implement filter
- I can handle collisions in a hash table
Closure
Closure Introduction
Closure
- Closure is the most esoteric of JavaScript concepts
- Enables powerful pro-level functions like ‘once’ and ‘memoize’
- Many JavaScript design patterns including the module pattern use closure
- Built iterators, handle partial application and maintain state in an asynchronous world
Functions get a new memory every run/invocation
1 | function multiplyBy2 (inputNumber) { |
- Function with memories
- When our functions get called, we created a live store of data(local memory/variable environment/state) for that function’s execution context.
- When the function finishes executing, its local memory is deleted(except the returned value)
- But what if our function could hold on to live data between executions?
- This would let our function definitions have an associated cache/persistent memory
- But it all starts with us returning a function from another function
Returning Functions
- Functions can be returned from other functions in JavaScript
1 | function createFunction() { |
Nested Function Scope
- Calling a function in the same function call as it was defined
1 | function outer () { |
Where you define your functions determines what data it has access to when you call it
Retaining Function Memory
- Calling a function outside of the function call in which it was defined
1 | function outer () { |
Function Closure
- 函数声明时会获得一个私有属性,一般为
[[scope]]
,里面保存了函数执行上下文所涉及到的变量环境(也就是函数的背包或者闭包的实际实现的方式)- 当函数使用上上级执行上下文是的变量,也会保存上上级执行上下文的变量环境(作用域链)
Closure Technical Definition & Review
-
- 在这本书中,作者将closure(闭包)称为Closed Over Variable Environment
- 闭包里保存的变量称为Persistent Lexically(or Static) Scope Reference Data
- 或者可以简单的称闭包为背包(Backpack)😂
每种语言都可以根据自己的需求制定自己的作用域规则,在JavaScript中,使用词法作用域(lexical Scope)或称静态作用域(Static Scope)的规则,既当函数定义时,就确定的函数所可以访问的变量范围。
- 同样,如果函数执行时才能确定访问的变量,称作动态作用域,具体区别可见这篇博客
What can we call this ‘backpack’
- Closed over ‘Variable Environment’(C.O.V.E.)
- Persistent Lexical Scope Referenced Data(P.L.S.R.D.)
- ‘backpack’
- ‘Closure’
The ‘backpack’(or ‘closure’) of live data is attached incrementCounter (then to myNewFunction) through a hidden property know as
[[scope]]
which persists when the inner function is returned out
Multiple Closure Instance
- Let’s run outer again
1 | function outer() { |
Practical Application
- Closure gives out functions persistent memories and entirely new toolkit for writing professional code
- Helper functions: Everyday professional helper functions like ‘once’ and ‘memorize’
- Iterators and generators: Which use lexical scoping and closure to achieve the most contemporary patterns for handling data in JavaScript
- Module pattern: Preserve state for the life of an application without polluting the global namespace
- Asynchronous JavaScript: Callbacks and promises rely on closure to persist state in an asynchronous environment
Closure Exercise
Asynchronous JavaScript
Single Threaded Execution Review
Promises, Async & Event Loop
- Promises - the most significant ES6 feature
- Asynchronicity - the feature that makes dynamic web applications possible
- The event loop - JavaScript triage
- Microtask queue, callback queue and Web Browser features(APIs)
A reminder of how JavaScript executes code
1 | const num = 3 |
Asynchronicity in JavaScript
Asynchronicity is the backbone of modern web development in JavaScript yet…
- JavaScript is:
- Single thread (one commend runs at a time)
- Synchronously executed (each line is run in order the code appears)
- So what if we have a task:
- Accessing Twitter’s server to get new tweets that takes a long time
- Code we want to run using those tweets
- Challenge: We want to wait for the tweets to be stored in tweets so that they’re there to run displayTweets on - but no code can run in the meantime
- JavaScript is:
Slow function blocks further code running
1 | const tweets = getTweets('http://twitter.com/will/1') |
- What if we try to delay a function directly using setTimeout?
- setTimeout is a built in function - its first arguments is to function to delay followed by ms to delay by
1 | function printHello() { |
- So what about a delay of 0ms
- Now, in what order will our console.log occur?
1 | function printHello() { |
- JavaScript is not enough - We need new pieces(some of which aren’t JavaScript at all)
- Our core JavaScript engine has 3 main parts:
- Thread of execution
- Memory/variable environment
- Call stack
- We need to add some new components:
- Web Browser APIs/Node background APIs
- Promises
- Event loop, Callback/Task queue and micro task queue
- Our core JavaScript engine has 3 main parts:
Asynchronous Browser Features
- ES5 solution: Introducing ‘callback functions’, and Web Browser APIs
1 | function printHello() { console.log('Hello') } |
Web API Rules
- We’re interacting with a world outside of JavaScript now - so we need rule
1 | function printHello() { console.log('hello') } |
Callback Queue & Event Loop
- 使用setTimeout或者setTimeout时,传入的回调函数在其他线程中开始计时,计时结束时推入callback queue, 此时代码不会阻塞线程,而是会继续执行,只有当前代码全部执行完毕,event loop才会开始轮询callback queue内是否有需要执行的回调函数
Callback Hell & Async Exercise
ES5 Web Browser APIs with callback functions
- Problems
- Our response data is only available in the callback function - Callback Hell
- Maybe it feels a little odd to think of passing a function into another function only for it to run much later
- Benefits
- Super explicit once you understand how it works under-the-hood
- Problems
Promises
Promises Introduction
- ES6+ Solution(Promises)
- Using two-pronged ‘facade’ functions that both:
- Initiate background web browser work and
- Return a placeholder object(promise) immediately in JavaScript
- Using two-pronged ‘facade’ functions that both:
Promises Example: fetch
- ES6+ Promises
1 | function display(data) { |
Promises Example: then
- 当我们使用Promise时,会会返回一个promise对象,其中对象会有一个隐藏的value属性代表新建promise时传入的回调函数的返回值,还有一个onFulfilled属性数组。当我们使用then方法是,就会将传入then方法的回调函数推入这个数组,在promise的状态变为resolved时,会将value作为参数传入onFulfilled数组中的函数依次执行。
Web APIs & Promises Example: fetch
- But we need to know how our promise-deferred functionality gets back into JavaScript to be run
1 | function display(data) { console.log(data) } |
then
method and functionality to call on completion- Any code we want to run on the returned data must also be saved on the promise object
- Added using
.then
method to the hidden propertyonFulfillment
- Promise objects will automatically trigger the attached function to run(with its input being returned data)
Web APIs & Promises Example: Microtask Queue
上面的代码中,在控制台打印’me first’之后会执行display而不是printHello,这是由于promise使用了microtask queue,而setTimeout使用callback queue(规范中称为task queue)
microtask queue的优先级是高于callback queue的,event loop会在当前执行线程完毕后首先检查microtask queue中是否有回调函数并执行,也就是将回调推入call stack,在microtask queue全部执行完后才会检查callback queue,所以在microtask queue中返回promise有可能造成死循环。而
callback queue执行完一个回调之后并不会阻塞线程,如果有未执行完毕的会推迟到下一个执行周期去执行、
Promises Review
Problems
- 99% of developers have no idea how they’re working under the hood
- Debugging becomes super-hard as a result
- Developers fail technical interviews
Benefits
- Cleaner readable style with pseudo-synchronous style code
- Nice error handling process
这里作者吐槽了then方法的命名有可能造成误解,应该命名为storeFunctionToRunLater😂,Promise应该命名为FutureData
如上所述,Promise的优点还有对错误处理的良好风格。
- Promise除了value、onFulfilled数组,还有一个onRejection数组,用来放置错误处理的回调函数。我们可以使用
.catch
方法或者在.then
方法中传入第二个参数来将回调添加到onRejection数组中。
- Promise除了value、onFulfilled数组,还有一个onRejection数组,用来放置错误处理的回调函数。我们可以使用
We have rules for the execution of our asynchronously delayed code
- Hold promise-deferred functions in a microtask queue and callback function in a task queue(Callback queue) when Web Browser Feature(API) finishes
- Add the function to the Call stack(i.e. run the function) when:
- Call stack is empty & all global code run (have the event loop check this condition)
- Prioritize functions in the microtask queue over the Callback queue
Promises, Web APIs, the Callback & Microtask Queues and Event loop enable:
- Non-blocking applications: This means we don’t have to wait in the single thread and don’t block further code from running
- However long it takes: We cannot predict when our Browser feature’s work will finish so we let JS handle automatically running the function on its completion
- Web applications: Asynchronous JavaScript is the backbone of the modern web - letting us build fast ‘non-blocking’ applications
Classes & Prototypes
Class & OOP Introduction
Classes, Prototypes - Object Oriented JavaScript
- An environment popular paradigm for structuring our complex code
- Prototype chain - the feature behind-the-scenes that enables emulation of OOP but is a compelling tool in itself
- Understanding the difference between
__proto__
and prototype - The new and class keywords as tools to automate our object & method creation
Core of development(and running code)
- Save data(e.g. in a quiz game the scores of user1 and user2)
- Run code(functions) using that data(e.g. increase user 2’s score)
- Easy! So why is development hard?
- In a quiz game I need to save lots of users, but also admins, quiz questions, quiz outcomes, league tables - all have data and associated functionality
- In 1000,000 lines of code
- Where is the functionality when I need it?
- How do I make sure the functionality is only used on the right data!
That is, I want my code to be:
- Easy to reason about, but also
- Easy to add feature to(new functionality)
- Nevertheless efficient and performance
The Object-oriented paradigm aims to let us achieve these three goals
Object Dot Notation
- So if I’m storing each user in my app with their respective data(let’s simplify)
– | User 1 | User 2 |
---|---|---|
name | Tim | Stephanie |
score | 3 | 5 |
And the functionality I need to have for each user(again simplifying!)
- Increment functionality(there’d be a ton of functions in practice)
How could I store my data and functionality together in one piece?
Objects - store functions with their associated data!
- This is the principle of encapsulation - and it’s going to transform how we can ‘reason about’ our code
1 | const user1 = { |
Let’s keep creating our objects. What alternative techniques do we have for creating objects?
Creating user2 use dot natation
- Declare an empty object and add properties with dot natation
1 | const user2 = {} // create an empty object |
- Creating user3 using
Object.create
- Object.create is going to give us fine-grained control over our object later on
1 | const user3 = Object.create(null) |
Our code is getting repetitive, we’re breaking our DRY principle. And suppose we have millions of users! What could we do?
Solution 1
- Generate objects using a function
1 | function userCreator(name, score) { |
- Solution 1. Generate Objects using a function
- Problems: Each time we create a new user we make space in our computer’s memory for all our data and functions. But our functions are just copies
- Is there a better way?
- Benefits: It’s simple and easy to reason about!
Prototype Chain
Solution 2: Using the prototype chain
- Store the increment function in just one object and have the interpreter, if it doesn’t find the function on user1, look up to that object to check if it’s there
- Link user1 and functionScore so the interpreter, on not finding
.increment
. makes sure to check up in functionStore where it would find it - Make the link with
Object.create()
technique
Solution 2: Using the prototype chain
1 | function userCreator(name, score) { |
Prototype Chain Example: Implicit Parameters
在上面的代码中,我们在声明increment方法时并不清楚会是user1还是user2调用此方法,那么我们如何确定我们需要增加分数的对象呢?
可以看到,这也正是
this
关键字的作用可以认为
this
指向方法调用时.
关键字前面的对象,如果没使用.
语法调用方法,可以认为this
指向全局对象
hasOwnProperty Method
- What if we want to confirm our user! has property score
1 | function userCreator(name, score) { |
We can use the
hasOwnProperty
method - but where is it?- Object.prototype.hasOwnProperty
What if we want to confirm our user1 has the property score?
- We can use the hasOwnProperty method - but where is it? Is it on user1?
- All objects have a
__proto__
property by default which defaults to linking to a big object - Object.prototype full of (somewhat) useful functions - We get access to it via userFunctionStore’s
__proto__
property - the chain
this Keyword
- Declaring & calling a new function inside our ‘method’ increment
1 | function userCreator(name, score) { |
Let’s start by simplifying(just increment method - written over 3 lines now)
Create and invoke a new function (add1) inside increment
1 | function userCreator(name, score) { |
- What does
this
get auto-assigned to?- window/global
Arrow Function Scope & this
- Arrow functions override the normal
this
rules
1 | function userCreator(name, score) { |
- Now our inner function gets its
this
set by where it was saved - it’s alexical scoped this
Prototype Chain Review
- Solution 2: Using the prototype chain
Problems: No problems! It’s beautiful. Maybe a little long-winded
Write this every single time - but it’s 6 words!
1
2
3const newUser = Object.create(userFunctionStore)
/** ... **/
return newUserBenefit: Super sophisticated but not standard
new Keyword
Solution 3 - Introducing the keyword that automates the hard work: new
- When we call the function that returns an object with
new
in front we automate 2 things- Create a new user object
- Return the new user object
- But now we need to adjust how we write the body of userCreator - how can we:
- Refer to the auto-created object?
this
- Know where to put our single copies of functions?
- prototype
- Refer to the auto-created object?
- When we call the function that returns an object with
The
new
keyword automates a lot of our manual work
1 | function userCreator(name, score) { |
- Interlude - functions are both objects and functions
1 | function multiplyBy2(num) { |
- We could use the fact that all functions have a default property ‘prototype’ on their object version.(itself an object) - to replace our ‘functionStore’ object
new Keyword Example
- The new keyword automates a lot of our manual work
1 | function userCreator(name, score) { |
class keyword
Solution 3 - Introducing the keyword that automates the hard work: new
- Benefits:
- Faster to write. Often used in practice in professional code
- Problem:
- 95% of developers have no idea how it works and therefore fail interviews
- We have to upper case first letter of the function so we know it requires
new
to work!
- Benefits:
Solution 4: The class ‘syntactic sugar’
- We’re writing our shared methods separately from our object ‘constructor’ itself(off in the
userCreator.prototype
object) - Other languages let us do this all in one place. ES2015 lets us do so too
- We’re writing our shared methods separately from our object ‘constructor’ itself(off in the
Solution 4: The class ‘syntactic sugar’
1 | class UserCreator { |
- Solution 4: The class ‘syntactic sugar’
- Benefit:
- Emerging as a new standard
- Feels more like style of other languages(e.g. Python)
- Problems:
- 99% of developers have no idea how it works and therefore fail interviews
- But you will not be one of them
- Benefit: