Chapter 1: Hello, world! | Learn Enough JavaScript to Be Dangerous | Learn Enough to Be Dangerous
You have to make a choice. Choose...wisely.

Get occasional notifications about things like product discounts, blog posts, and new or updated tutorials. Unsubscribe at any time.

Quick Checkout
or Pay by Credit Card
Error processing your payment
  • You didn't choose whether or not to be added to the mailing list
Confirm
$0.00

Payments and credit card details are securely managed and protected by Learn Enough's payment processor, Stripe. More information on their site:

CART
Total
$0.00

Your Cart is Empty

$30
$300
$300
$XY
$XY
1234
Get Single Tutorial
MORE INFO

Learn Enough JavaScript to Be Dangerous is available as an ebook, an offline video series, and as a structured, self-paced online course. The course includes full online access to the book content, streaming videos, progress tracking, exercises, and community exercise answers.

All Access Subscription
MORE INFO

The Learn Enough All Access Subscription includes the entire Learn Enough introductory sequence and the full Ruby on Rails Tutorial. More than 2500 pages of book content and 53 hours of video that teach you to code from total beginner up to professional-grade web development.

Sign up for the course and get access to the full tutorial and streaming screencasts!

Learn Enough JavaScript to Be Dangerous Write Programs, Publish Packages, and Develop Interactive Websites with JavaScript Michael Hartl

Write Programs, Publish Packages, and Develop Interactive Websites with JavaScript

Learn Enough JavaScript to Be Dangerous

A tutorial introduction to programming with JavaScript

Michael Hartl

Contents

About the author

Michael Hartl is the creator of the Ruby on Rails Tutorial, one of the leading introductions to web development, and is cofounder and principal author at Learn Enough. Previously, he was a physics instructor at the California Institute of Technology (Caltech), where he received a Lifetime Achievement Award for Excellence in Teaching. He is a graduate of Harvard College, has a Ph.D. in Physics from Caltech, and is an alumnus of the Y Combinator entrepreneur program.

Chapter 1 Hello, world!

As the only language that can be executed inside web browsers, JavaScript is an essential part of every programmer’s toolkit. Learn Enough JavaScript to Be Dangerous is designed to get you started writing practical and modern JavaScript programs as fast as possible, using the latest JavaScript technologies (including Node.js and ES6), with a focus on the real tools used every day by software developers.

Unlike most JavaScript tutorials, we’ll be treating JavaScript as a general-purpose programming language right from the start, so our examples won’t be confined to the browser. The result is a practical narrative introduction to JavaScript—a perfect complement both to in-browser coding tutorials and to the voluminous but hard-to-navigate JavaScript reference material on the Web.

You won’t learn everything there is to know about JavaScript—that would take thousands of pages and centuries of effort—but you will learn enough JavaScript to be dangerous (Figure 1.1).1

images/figures/flavian_amphitheater
Figure 1.1: JavaScript knowledge, like Rome, wasn’t built in a day.

There are no programming prerequisites for Learn Enough JavaScript to Be Dangerous, although it certainly won’t hurt if you’ve programmed before. What is important is that you’ve started developing your technical sophistication (Box 1.1), either on your own or using the preceding Learn Enough tutorials. These tutorials include the following, which together make a good list of prerequisites for this book:

  1. Learn Enough Command Line to Be Dangerous
  2. Learn Enough Text Editor to Be Dangerous
  3. Learn Enough Git to Be Dangerous
  4. Learn Enough HTML to Be Dangerous
  5. Learn Enough CSS & Layout to Be Dangerous
Box 1.1. Technical sophistication

An essential aspect of using computers is the ability to figure things out and troubleshoot on your own, a skill we at Learn Enough call technical sophistication.

Developing technical sophistication means not only following systematic tutorials like Learn Enough Command Line to Be Dangerous, Learn Enough Git to Be Dangerous, Learn Enough HTML to Be Dangerous, and Learn Enough CSS & Layout to Be Dangerous, but also knowing when it’s time to break free of a structured presentation and just start Googling around for a solution.

Learn Enough JavaScript to Be Dangerous will give us ample opportunity to practice this essential technical skill.

In particular, as alluded to above, there is a wealth of JavaScript reference material on the Web, but it can be hard to use unless you already know basically what you’re doing. One goal of this tutorial is to be the key that unlocks the documentation. This will include lots of pointers to my favorite JavaScript source, the Mozilla Developer Network (MDN) Web Docs (or just “MDN” for short).

Especially as the exposition gets more advanced, I’ll also frequently include the exact web searches I used to figure out how to accomplish the particular task at hand. For example, how do you use JavaScript to return all elements on a page that match, say, a particular CSS class? Like this: Google “javascript css class return all elements”.

In order to learn enough JavaScript to be dangerous, we’ll begin at the beginning with a series of simple “hello, world” programs using several different techniques (Chapter 1), including an introduction to Node.js, a fast and widely used execution environment for JavaScript programs. In line with the Learn Enough philosophy of always doing things “for real”, even as early as Chapter 1 we’ll deploy a (very simple) dynamic JavaScript application to the live Web.

After mastering “hello, world”, we’ll take a tour of some JavaScript objects, including strings (Chapter 2), arrays (Chapter 3), and other native objects (Chapter 4). Taken together, these chapters constitute a gentle introduction to object-oriented programming with JavaScript.

In Chapter 5, we’ll learn the basics of functions, an essential subject for virtually every programming language. We’ll then apply this knowledge to an elegant and powerful style of coding called functional programming (Chapter 6).

Having covered the basics of built-in JavaScript objects, in Chapter 7 we’ll learn how to make objects of our own. In particular, we’ll define an object for a phrase, and then develop a method for determining whether or not the phrase is a palindrome (the same read forward and backward).

Our initial palindrome implementation will be rather rudimentary, but we’ll extend it in Chapter 8 using a powerful technique called test-driven development (TDD). In the process, we’ll learn more about testing generally, as well as how to create and publish a self-contained software package called an NPM module (and thereby join the large and growing ecosystem of software packages managed by npm, the Node Package Manager).

In Chapter 9, we’ll apply our new NPM module to a JavaScript web application: a site for detecting palindromes. This will give us a chance to learn about events and DOM manipulation. We’ll start with the simplest possible implementation, and then add several extensions of steadily increasing sophistication, including alerts, prompts, and an example of an HTML form.

In Chapter 10, we’ll learn how to write nontrivial shell scripts using JavaScript, a much-neglected topic that underscores JavaScript’s growing importance as a general-purpose programming language. Examples include reading from both files and URLs, with a final example showing how to manipulate a downloaded file as if it were an HTML web page.

In Chapter 11, we’ll apply the techniques from Chapter 9 and Chapter 10 to a real, industrial-grade website. In particular, we’ll extend the sample application from Learn Enough CSS & Layout to Be Dangerous to add a functional image gallery that dynamically changes images, CSS classes, and page text in response to user clicks. (We’ll be using Git to clone a repository directly, so you’ll be able to build and deploy the image gallery even if you haven’t completed Learn Enough CSS & Layout to Be Dangerous.)

In most cases, typing in code examples by hand is the most effective way to learn, but sometimes copying and pasting is more practical. To make the latter more convenient, all code listings from this book are available online at the following URL:

https://github.com/learnenough/learn_enough_javascript_code_listings

Although full-blown web development with a dynamically rendered front-end and a database back-end is beyond the scope of this book, by the end of Learn Enough JavaScript to Be Dangerous you’ll have a solid foundation on which to build such skills. We’ll end the tutorial with pointers to additional resources for extending your JavaScript knowledge further, as well as to further Learn Enough tutorials for full-stack web development—specifically, using Ruby (via Sinatra) and Ruby on Rails, for which a background in JavaScript is excellent preparation.

1.1 Introduction to JavaScript

JavaScript was originally developed by computer scientist Brendan Eich (Figure 1.2)2 for Netscape Navigator, the first commercial web browser, under the name “LiveScript” (Box 1.2). The original use of JavaScript is still its main use—namely, “making cool things happen on web pages”, typically via manipulation of the Document Object Model (DOM) introduced in Learn Enough CSS & Layout to Be Dangerous. In recent years, though, JavaScript’s role has expanded significantly, and it is now often used as a back-end and general-purpose programming language as well.

brendan_eich
Figure 1.2: Brendan Eich, the creator of JavaScript.
Box 1.2. What’s in a name?

What’s in a name? That which we call a rose By any other word would smell as sweet.

—William Shakespeare, Romeo and Juliet 2.2.46–47

What we now call JavaScript was originally called “LiveScript” by its creators at Netscape, but at the time of its planned release there was an enormous amount of hype about Java, a language developed by Sun Microsystems. In an attempt to capitalize on this hype, Netscape changed LiveScript’s name to “JavaScript”—thereby causing endless confusion for developers wondering what it has to do with Java. (The answer is: nothing.)

Later on, a standardized version of JavaScript called ECMAScript (pronounced ECK-muh-script) was created in an effort to improve cross-browser compatibility. Technically, what most people call “JavaScript” is more properly called “ECMAScript”, with JavaScript being only ECMAScript’s most common implementation, but in this tutorial we follow the common convention of using “JavaScript” to refer to the language in general. The main exception to this rule is our occasional use of contracted names like “ES6”, which refers to the sixth edition of ECMAScript (a particularly large and important update, adding many useful features to the ECMAScript/JavaScript standard).

Finally, it’s worth noting that the misspelling “Javascript”, with a lowercase “s”, is extremely common, to the point of being borderline acceptable, even in relatively formal contexts. I frankly find the “Javascript” spelling to be more appealing than the rather pedantic official version, but it is technically wrong, so in this tutorial we’ll stick with “JavaScript”, and be technically correct.

In order to give you the best broad-range introduction to programming with JavaScript, Learn Enough JavaScript to Be Dangerous uses four main methods:

  1. Front-end JavaScript programs running in the user’s browser
  2. An interactive prompt with a Node.js Read-Evaluate-Print Loop (REPL)
  3. Standalone JavaScript files (including the Node Package Manager)
  4. Shell scripts (as introduced in Learn Enough Text Editor to Be Dangerous)

We’ll begin our study of JavaScript with four variations on the time-honored theme of a “hello, world” program, a tradition that dates back to the early days of the C programming language. The main purpose of “hello, world” is to confirm that our system is correctly configured to execute a simple program that prints the string hello, world! (or some close variant) to the screen. By design, the program is simple, allowing us to focus on the challenge of getting the program to run in the first place.

Since the original and still most common application of JavaScript is to write programs that execute on the Web, we’ll start by writing (and deploying!) a program to display a greeting in a web browser. We’ll then write a series of three programs using the JavaScript execution system Node.js: first in the Node REPL, then in a JavaScript library file called hello.js, and finally in an executable shell script called hello.

Throughout what follows, I’ll assume that you have access to a Unix-compatible system like macOS or Linux (including the Linux-based Cloud9 IDE, as described in the free tutorial Learn Enough Dev Environment to Be Dangerous). If you use the cloud IDE, I recommend creating a development environment called javascript-tutorial. The cloud IDE uses the Bash shell program by default, and Mac users can use whichever shell program they prefer—this tutorial should work with either Bash or the default Z shell (Zsh). See “Using Z Shell on Macs with the Learn Enough Tutorials” for more information.

1.2 JS in a web browser

Even though JavaScript is increasingly used as a general-purpose programming language, it still thrives in its native habitat of the web browser. Accordingly, our first “hello, world” program involves displaying a notification, or alert, created by JavaScript code on a web page.

We’ll begin by making a directory for this tutorial using mkdir -p (which creates intermediate directories as necessary),3 along with an HTML index file using the touch command:4

$ mkdir -p ~/repos/js_tutorial
$ cd ~/repos/js_tutorial
$ touch index.html

Next, we’ll follow the practice introduced in Learn Enough Git to Be Dangerous and put our project under version control with Git:

$ git init
$ git add -A
$ git commit -m "Initialize repository"

At this point, we’re ready to make our first edit. We’ll start in familiar territory by adding a simple HTML skeleton (without JavaScript) to our index page, as shown in Listing 1.1. The result appears in Figure 1.3.

Listing 1.1: An HTML skeleton. index.html
<!DOCTYPE html>
<html>
  <head>
    <title>Learn Enough JavaScript</title>
    <meta charset="utf-8">
  </head>
  <body>
    <h1>Hello, world!</h1>
    <p>This page includes an alert written in JavaScript.</p>
  </body>
</html>
images/figures/html_skeleton
Figure 1.3: Our initial static index page.

This page’s paragraph is a little lie, because we haven’t yet added any JavaScript. Let’s change that by putting in a script tag containing a single command:

<script>
  alert("hello, world!");
</script>

Here we’ve used alert, which is a JavaScript function, a piece of code that takes in arguments and performs some task with them. As shown in Figure 1.4, the anatomy of a JavaScript function call is the function’s name, an open parenthesis, zero or more arguments, a closing parenthesis, and a semicolon to end the line. (We’ll learn more about functions, including how to define our own, in Chapter 5.)

images/figures/function_anatomy
Figure 1.4: The anatomy of a JavaScript function call.

In this case, alert takes in a string (Chapter 2) and displays it as an alert in the browser. To see this in action, let’s add the alert code to our index page, as shown in Listing 1.2. Technically, we could place the script tag anywhere on our page, but it’s conventional to place it in the head of the document (especially when including external JavaScript files, as we’ll see in Section 5.2).

Listing 1.2: “Hello, world!” in JavaScript. index.html
<!DOCTYPE html>
<html>
  <head>
    <title>Learn Enough JavaScript</title>
    <meta charset="utf-8">
    <script>
      alert("hello, world!");
    </script>
  </head>
  <body>
    <h1>Hello, world!</h1>
    <p>This page includes an alert written in JavaScript.</p>
  </body>
</html>

Upon refreshing the page, our browser now displays a friendly greeting (Figure 1.5).

images/figures/js_hello_world
Figure 1.5: The result of a “hello, world!” alert.

1.2.1 Deployment

As a final step, let’s deploy our incredibly fancy JavaScript app to the live Web. Our technique is the same one covered in Learn Enough Git to Be Dangerous, Learn Enough HTML to Be Dangerous, and Learn Enough CSS & Layout to Be Dangerous, namely, a free site hosted at GitHub Pages.

Deploying at even this early stage is a powerful proof-of-concept—all kidding aside about our “incredibly fancy” app, we really are deploying a live website, which was an enormously difficult step only a few years ago, and yet now we can do it in seconds.

First, let’s commit the changes made in Listing 1.2:

$ git commit -am "Add a JavaScript 'hello, world'"

The next step is to create a new remote repository at GitHub, as shown in Figure 1.6. (If any of these steps are unfamiliar, consult Learn Enough Git to Be Dangerous for details.)

images/figures/new_repository
Figure 1.6: Creating a new repository at GitHub.

Next, configure your local system with the remote repository and push it up (taking care to fill in <username> with your GitHub username and using a GitHub personal access token when prompted for a password) and then push it up:

$ git remote add origin https://github.com/<username>/js_tutorial.git
$ git push -u origin main

Because videos are relatively hard to update, the screencasts that accompany this book use master, which was the default branch name for the first 15+ years of Git’s existence, but the text has been updated to use main, which is the current preferred default. See the Learn Enough blog post “Default Git Branch Name with Learn Enough and the Rails Tutorial” for more information.

To complete the deployment, all we need to do is edit the Settings (Figure 1.7) and configure our site to be served off the main branch by GitHub Pages, as shown in Figure 1.8 and Figure 1.9.

images/figures/github_settings
Figure 1.7: Editing the settings for a GitHub repository.
images/figures/github_pages_main_branch
Figure 1.8: Serving our website from the main branch.
images/figures/github_pages_save
Figure 1.9: Saving the new GitHub Pages settings.

With that, you can now visit your site at the following URL:5

https://<username>.github.io/js_tutorial

The result should be the same “hello, world!” greeting seen in Figure 1.5, except now on the live Web (Figure 1.10).It’s alive!” (Figure 1.11).6

images/figures/js_hello_world_live
Figure 1.10: A JavaScript “hello, world!” page on the live Web.
images/figures/frankenstein
Figure 1.11: Bringing a site to life is easier than it used to be.

1.2.2 Exercises

  1. What happens if you put a second alert after the first one?

1.3 JS in a REPL

Our next two closely related examples of a “hello, world” program involve a Read-Eval-Print Loop, or REPL (pronounced “repple”). A REPL is a program that reads input, evaluates it, prints out the result (if any), and then loops back to the read step. Most modern programming languages provide a REPL, and JavaScript is no exception. In fact, as hinted above, it actually provides two.

1.3.1 Browser console

Our first example of a REPL is the browser console, which is available in most modern browsers as part of the standard suite of developer tools. Whether these tools are available by default depends on the browser you use; they’re included automatically in Google Chrome, for example, but in Safari they have to be installed. Use your technical sophistication (Box 1.1) to figure out the setup for your browser of choice.

The developer tools can typically be accessed by right-clicking (or Ctrl-clicking) in your browser window and selecting Inspect Element to open the web inspector (Figure 1.12). The result should look something like Figure 1.13.

images/figures/inspect_element
Figure 1.12: Activating the developer tools via Inspect Element.
images/figures/developer_tools
Figure 1.13: The browser developer tools.

At this point, we’re ready to access the console by clicking on the corresponding tab in the developer tools, as shown in Figure 1.14. As we’ll see in Section 5.2, the console is a valuable debugging tool, as it has access to the full DOM and other aspects of our application’s environment, as well as displaying any warnings or errors that might affect our application. In particular, note that Figure 1.14 shows a warning (regarding a missing favicon.ico file); knowing when you can and can’t safely ignore such warnings is a hallmark of technical sophistication. (In this case, it’s safe to ignore. In addition, your setup may or may not show the exact same error. Is that a problem?)

images/figures/console
Figure 1.14: The interactive JavaScript console.

We’re finally ready to write our “hello, world” program using the console REPL. Our method is to use console, which is a JavaScript object that represents the console and its associated data, functions, etc. In particular, the console object has a function called log, which prints out (“logs”) its argument to the screen. We can access it using a “dot” notation that has become standard across a wide variety of object-oriented languages, as seen in Listing 1.3.

Listing 1.3: A “hello, world” command in the console.
> console.log("hello, world!");

In the context of objects, a function like log called using the dot notation is often called a method.

At this point, you should type the console.log command into your browser console, noting that the > in Listing 1.3 represents the console prompt itself, and shouldn’t be typed literally. The result should resemble Figure 1.15. (We’ll explain the meaning of undefined in Section 2.3.)

images/figures/console_hello_world
Figure 1.15: Printing out “hello, world!” in the browser console.

Alert readers (no pun intended) might have noticed that the command in Listing 1.3 includes a terminating semicolon (Figure 1.4), whereas the command shown in Figure 1.15 doesn’t. This discrepancy is included in order to show that the two commands work the same, and it is common to omit the semicolon when using an interactive console. For consistency, we’ll generally include the semicolon throughout the rest of this tutorial (even in consoles), but it’s good to be aware of both conventions in case you see something different in other people’s code.

1.3.2 Node prompt

Every web browser in the known Universe can execute JavaScript programs, but part of treating JavaScript as a general-purpose programming language means running it at the command line as well. This means installing and using a command-line program capable of evaluating JavaScript programs, and nowadays the most popular choice is abundantly clear: Node.js (usually pronounced “node jay-ess”, and often called “Node” for short).

It’s possible that Node.js is already installed on your system. The easiest way to check is to use the which command (as described in Learn Enough Command Line to Be Dangerous):

$ which node
/usr/local/bin/node

If the path to a node executable is displayed, you’re good to go.

If Node isn’t present on your system, you should install it at this point. If you’re using a Macintosh with Homebrew installed, you can run

$ brew install node

to get the latest version. If you already have it installed, run

$ brew upgrade node

instead.

Otherwise, go to the Node.js website and follow the download and installation instructions for your system.

Once it’s installed, running the Node.js REPL is easy—just run the node command at the command line, as seen in Listing 1.4.

Listing 1.4: Bringing up the Node prompt at the command line.
$ node
>

As with the browser console, > represents the Node prompt, and like the console it allows us to run commands interactively. (For simplicity, we’ll sometimes use “console” to refer either to the browser console or to the Node REPL.) In particular, to replicate the “hello, world” program from Listing 1.3, we can simply type the same command at the Node prompt, as follows:

> console.log("hello, world!");
hello, world!

(Depending on your system, you might see undefined appear as well; we’ll discuss this detail in Section 2.3.)

That’s it! In both the browser console and Node prompt, we can print “hello, world!” with this single command:

> console.log("hello, world!");

1.3.3 Exercises

  1. What happens if you run an alert in the browser console? What about in the Node console?

1.4 JS in a file

As convenient as it is to be able to explore JavaScript interactively, most Real Programming™ takes place in text files created with a text editor. In this section, we’ll show how to create and execute a JavaScript file with the same “hello, world” program we’ve discussed in the previous two sections. The result will be a simplified prototype of the reusable JavaScript files we’ll start learning about in Section 5.2.

We’ll start by creating a JavaScript file (with a .js file extension) for our hello program:

$ touch hello.js

Next, using our favorite text editor, we’ll fill the file with the contents shown in Listing 1.5. Note that the code is exactly the same as in Listing 1.3 and subsequent examples, with the difference being that in a JavaScript file there’s no command prompt >.

Listing 1.5: A “hello, world” program in a JavaScript file. hello.js
console.log("hello, world!");

At this point, we’re ready to execute our program using the same node command we used in Listing 1.4 to bring up the Node prompt. The only difference is that this time we include an argument with the name of our file:

$ node hello.js
hello, world!

As before, the result is to print “hello, world!”, this time to the terminal screen. (Inside the program, the return value of console.log is undefined as before, but it’s not displayed since, unlike with interactive prompts, return values aren’t displayed by command-line programs.)

Although this example is simple, it’s a huge step forward, as we’re now in the position to write JavaScript programs much longer than could comfortably fit in an interactive console or Node session.

1.4.1 Exercises

  1. What happens if you give console.log two arguments, as in Listing 1.6?
Listing 1.6: Using two arguments. hello.js
console.log("hello, world!", "how's it going?");

1.5 JS in a shell script

Although the code in Section 1.4 is perfectly functional, when writing a program to be executed in the command line shell it’s often better to use an executable script of the sort discussed in Learn Enough Text Editor to Be Dangerous. Now that JavaScript can be used so effectively outside the browser, it has joined more traditional “scripting languages” like Perl, Python, and Ruby as an excellent choice for writing such shell scripts.

Let’s see how to make an executable script using Node. We’ll start by creating a file called hello:

$ touch hello

Note that we didn’t include the .js extension—this is because the filename itself is the user interface, and there’s no reason to expose the implementation language to the user. Indeed, there’s a reason not to: by using the name hello, we give ourselves the option to rewrite our script in a different language down the line, without changing the command our program’s users have to type. (Not that it matters in this simple case, but the principle should be clear. We’ll see a more realistic example in Section 10.3.)

There are two steps to writing a working script. The first is to use the same command we’ve seen before (Listing 1.5), preceded by a “shebang” line telling our system to use node to execute the script.

The exact shebang line is system-dependent; you can find the proper executable path for your system by running the which command:

$ which node
/usr/local/bin/node

Using this command for the shebang line in the hello file gives the shell script shown in Listing 1.7.

Listing 1.7: A “hello, world” shell script. hello
#!/usr/local/bin/node

console.log("hello, world!");

We could execute this file directly using the node command as in Section 1.4, but a true shell script should be executable without the use of an auxiliary program. (That’s what the shebang line is for.) Instead, we’ll follow the second of the two steps mentioned above and make the file itself executable using the chmod (“change mode”) command combined with +x (“plus executable”):

$ chmod +x hello

At this point, the file should be executable, and we can execute it by preceding the command with ./, which tells our system to look in the current directory (dot = .) for the executable file. (Putting the hello script on the PATH, so that it can be called from any directory, is left as an exercise.) The result looks like this:

$ ./hello
hello, world!

Success! We’ve now written a working JavaScript shell script suitable for extension and elaboration. As mentioned briefly above, we’ll see an example of a real-life utility script in Section 10.3.

Throughout the rest of this tutorial, we’ll mainly use the Node REPL for initial investigations, but the eventual goal will almost always be to create a file (either pure code or HTML) containing JavaScript.

1.5.1 Exercises

  1. By moving the file or changing your system’s configuration, add the hello script to your environment’s PATH. (You may find the steps in Learn Enough Text Editor to Be Dangerous helpful.) Confirm that you can run hello without prepending ./ to the command name.

Join the Mailing List

Get occasional notifications about things like product discounts, blog posts, and new or updated tutorials. Unsubscribe at any time.