Skip to content

Series: JavaScript the Good Parts - Chapter 2 - Grammar - Review

Posted on:April 30, 2020 at 12:00 AM (7 min read)
test

As we continue reading JavaScript: The Good Parts by Douglas Crockford - I continue taking notes and updating my blog posts!

Chapter 2 focuses on JavaScript’s Grammar, which is very important to know while writing it :).

Of course, he maintains the focus only on parts he believes as good, so it won’t be a thorough view.

Whitespace

There are rarely any requirements for whitespace; but, you will find you need to separate sequences like declaring a variable.

As developers, we typically add whitespace to help improve readability. We then add a build process that takes that nice, beautifully formatted readable code to uglify it or minify it to reduce the file sizes when loaded in the browser.

I say this because knowing when and where JS requires whitespace isn’t a key to knowing the language. Our IDE should take care of it for us.

Comments

There are two different ways to create comments in JS:

Line Comments:

// this line would be commented

Block Comments:

/*
  This block 
  of
  would be commented
*/

Mr. Crockford recommends only using //, so we can assume he never writes method headers or class headers in his codebases.

Joking aside. He says this due to regular expressions creating errors with block comments - maybe he does this often, but in my experience, code bases don’t have regular expressions all over the place.

I don’t agree with this statement at all for two reasons:

  1. Comments in code are usually to give context to why code is written a specific way, not to comment the code out (commented code should be removed from codebases)
  2. Method and class headers with JSDoc syntax uses block commenting by default, especially when IDEs help create the blocks for you.

Names

The book defines a name as “a letter optionally followed by one or more letters, digits, or underbars.”

This is incorrect as you can name things starting with underbars.

const _test = 'hi' works fine.

MDN variable section states the following:

“A JavaScript identifier must start with a letter, underscore (_), or dollar sign ($). Subsequent characters can also be digits (0–9).”

I didn’t look into when this changed but I do think it’s important to know.

Names in JS are case sensitive, so these are not the same.

const test = 'test';
const TEST = 'hi';

console.log(test, TEST); // test hi

Reserved word list (ES6):

arguments
await
break
case
catch
class
const
continue
debugger
default
delete
do
else
enum
eval
export
extends
false
finally
for
function
if
implements
import
in
instanceof
interface
let
new
null
package
private
protected
public
return
static
super
switch
this
throw
true
try
typeof
var
void
while
with
yield

He does mention that in addition to reserved words not being allowed in variable names, object properties can’t be either. In Chrome dev console, I’m easily able to create an object:

const test = {class: 'hi'}

I looked into the ECMA Script standard for more detail here; unfortunately I didn’t find anything around object properties. If you find or know anything about this - please let me know in the comments!

I did find it interesting that he mentions a few words that aren’t reserved but should have been, such as undefined, NaN, and infinity. That is very surprising to me, and I’m thankful it was pointed out.

You can use - to denote negative numbers and e for exponents.

console.log(1e2); //100

Numbers

”JavaScript has a single number type. Internally, it is represented as a 64-bit floating point, the same as Java’s double.

BigInts are now available!

console.log(3 === 3.0) // true

In most other languages, this statement would be false.

Douglas Crockford believes that this creates a lot of convenience for us because we don’t have to worry about automatic type conversions in numbers or worry about overflowing integer values.

NaN means Not a Number, represents in JS when you try an operation between numbers that isn’t allowed.

In the past, we used isNaN as a global function for checking for this value. It had some weird rules and caused issues, so a new function was created: Number.isNaN which can be learned about here.

Lastly, we use the Math class for common math operations between numbers in JS.

Strings

”String literals can be wrapped in single quotes or double quotes.”

Additionally, template literals were introduced and are convenient for building long strings with string interpolation.

Each string can contain 0+ characters. You can use \ to escape characters.

JS was built when Unicode was a 16-bit charset, so all characters are 16 bits wide. Further, there are no char types, but you can use a string of one character instead.

”Strings are immutable, a string can never be changed. But it is easy to make a new string by concatenating other strings together…”

You can use either concatenation or template literals to join strings.

// initialization
const test = "test";
const test1 = "test1";
const test2 = `test2`;

// concatenation
const test3 = "test3" + "test3";

// interpolation
console.log(`${test} ${test1} ${test2} ${test3}`);
// test test1 test2 test3 test3

// escaping (example from the book)
"A" === "\u0041";

I prefer single quotes for initialization - they are less intrusive unless utilizing string templates for interpolation. Interpolation is a bit slower than string concatenation, but also cleaner.

Two strings are equal in JS if they are exactly the “same characters in the same order.”

Lastly - Strings in JS have methods. Technically these methods are on the String object class, not string literals, but thankfully for us devs, JS converts them for us.

Statements

”A compilation unit contains a set of executable statements.”

There is a list of updated statements on MDN.

I’d rather not dive deep into each of these, as many of them are fundamental programming topics - but will note a few things.

Variable scope

In the past, JavaScript only had variable declarations using var. The scope of variables declared with var is the function it is declared in, not the block. Two new types, let and const declarations were introduced to add block-level scoping.

Truthy and Falsy

With JavaScript being loosely typed, it coerces different variables into types when evaluating statements.

For example, an if statement.

const test = { hi: 'hi' };

if (test) {
 console.log('it is truthy!');
} else {
 console.log('it is falsy!');
}

When evaluating this statement, JS must determine whether test is true or false, but we can see it’s an object, not a boolean.

In JS, only the following values are falsy:

false
null
undefined
''
0
NaN

All other values are truthy.

The example above prints it is truthy! since test isn’t one of the falsy values.

Expressions

This section of the book goes into the details of all the different types of operators and expressions in JS. I felt that they were poorly explained, so naturally, MDN came to the rescue.

Definition from MDN:

“An expression is any valid unit of code that resolves to a value."

"Every syntactically valid expression resolves to some value but conceptually, there are two types of expressions: with side effects (for example: those that assign value to a variable) and those that in some sense evaluate and therefore resolve to a value.”

Type 1 (an expression with side effect): const name = 'Kaleb'

Type 2 (an expression that evaluates and resolves to a value) (3 + 3)

There are five expression categories:

  1. Arithmetic - evaluating to a number, typically using JS arithmetic operators like + or -
  2. String - expression evaluates to a string
  3. Logical - resolves to true or false utilizing logical operators in many cases
  4. Primary expressions - basic keywords and general expressions
  5. Left-hand-side expressions - “left values are the destination of an assignment”

The MDN link above goes into more depth on the different operators and expression categories.

Lastly, check out the reference for operator precedence - the order at which JS evaluates an expression’s operators.

He ends this chapter with two additional sections on objects and functions, but each has a chapter dedicated to them. Because of this, I will include them in the future for articles on those chapters.