Objects and prototype chain

Learning Goals

By the end of this lesson, you will be able to:

  • Speak to and understand the difference between prototoype and __proto__
  • Understand and explain how prototypal inheritance works via the prototype chain.

Vocab

  • prototype A special object which is assigned to functions that you make in JavaScript
  • [[prototype]] A hidden link on every object that links objects to one another, allowing objects to share behaviors
  • __proto__ Also called “dunder-proto”, this allows us to access the [[prototype]]

Prototype

Warm Up

Where have we heard the term prototype before. What do we know about this so far?

Review

  • We know that a prototype is a model of something and how it should look or behave.
  • We know that we have access to prototype methods (forEach, filter, etc) on arrays.

What is a prototype?

In JavaScript, a prototype is a special object that is assigned to all functions, including but not limited to functions that are used to construct objects.

Take a look at the example below:

const createInstructor = (name, primaryLesson) => {
  return {
    name: name,
    primaryLesson: primaryLesson,
    teachLesson: (duration) => {
      if (duration > 3) {
        return `${this.name} can\'t teach a lesson that long!`
      } else {
        return `Gunna teach you all real good about ${this.primaryLesson}`
      }
    },
    gradeProjects: (project) => {
      return `${this.name} is grading ${project}.`
    } 
  }
}

const nick = createInstructor('Nick', 'Intro to Accessibility')
  • After declaring this function and initializing the variable nick, type nick. into your console. You should see the browser attempting to autocomplete this with properties that are available on this function. Look at the nick.constructor.prototype property and take note of its value.

Diving Deeper

Let’s consider the Array data type in JavaScript. In your console, access the prototype of Array - what do you notice? The constructor of all arrays defines the methods on our Array.prototype - you might be familiar with some of these array prototype methods already!

Our createInstructor function which creates new instructor objects is also known as a constructor function. In object-oriented programming, which is based on classes, developers use constructor functions to create multiple objects from a template. In legacy React code, it is not uncommon to encounter classes. Here is the some code written using ES5 class syntax:

function Instructor(name, primaryLesson) {
    this.name = name;
    this.primaryLesson = primaryLesson;
}

Instructor.prototype.teachLesson = function(duration) {
  if (duration > 3) {
    return `${this.name} can\'t teach a lesson that long!`;
  } else {
    return `Gunna teach you all real good about ${this.primaryLesson}`;
  } 
}

Instructor.prototype.gradeProjects = function(project) {
  return `${this.name} is grading ${project}.` 
}

let instructor = new Instructor('Robbie', 'POST Requests');

In order to add methods to our template, we had to manipulate the prototype directly. We can do with with our Array type. First, we will define a new prototype method on Array, and then access that method like we would for .map() or .filter().

Array.prototype.getLast = function() {
  return this[this.length - 1]
}

const ingredients = ['sauce', 'cheese', 'crust', 'peppers']

ingredients.getLast()

What is this referring to in the example above?

Review: What is a prototype?

A special object which is assigned to functions that you make in JavaScript

Another Look (In Breakout Groups)

From our prior example, take a look at the nick and take note of all the properties. (especially the prototype property)

instance in the console

It is likely that you have seen this (somewhat overwhelming) list of properties while working with other types of data in your console - not just functions and objects.

  • After doing the above, declare an array literal named food in your console. (list the last three things you ate)
  • What happens when you type food.?
  • What is the browser attempting to autocomplete?
  • What familiar properties are showing up?
  • Is the prototype property still there? Why or why not?
  • Bonus: Try this out with other data types!

proto

You’ll notice that when we checked the instructor instance and array literal in the console, there was no property of prototype listed. This is because neither of these are functions, so the property of prototype is not automatically assigned.

What is the proto?

Everything that we have checked in the console thus far has had the property __proto__.

  • __proto__, or “dunder-proto” (double-underscore proto), is a property that exposes the internal [[prototype]] property.
  • This [[prototype]]property acts a as a reference that points to a function’s prototype object.

dunder proto at work

Note

It is this reference, or linking, between objects that makes up the prototype chain.

  • The prototype chain (and objects linking in such a way) is what allows us to define behavior in one place (or one prototype object) and share it between many objects.

For example, using our instructor variable, let’s use the valueOf prototype method.

// Example #1
let instructor = createInstructor('Travis', 'OOP');

instructor.valueOf()
// {instructor: "Travis", module: 2}

// Example #2
let message = 'The prototype chain is super cool!';
message.valueOf();
// "The prototype chain is super cool!"

Strange…this isn’t a method we created on our createInstructor constructor method. valueOf is actually an object prototype method! Why do we have access to this method on a string as well? What is happening?!

When the interpreter is attempting to access a property on an object, it will start by looking directly on that object. If that property is not found, the objects internal [[prototype]] property will point to the object to examine next. The interpreter will continue to travese the prototype chain to seek a resolution for the property… until the end of the prototype chain is reached (resulting in null).

It is this linking between objects that makes it so that objects are not limited to using the methods that are formally defined directly on that object - they can also ask objects further up the chain for help.

the prototype chain simplified

In Breakout Groups

With your partner, save all the data types that you can think of to variables. Using the console, check the [[prototype]] link via the dunder-proto.

  • What similiarties do you notice across data types? What differences?
  • Can we say “everything is an object” in JavaScript?

Checks for Understanding

  • Describe the prototype chain. Why is the chain in place? Why would the interpreter traverse this chain?
  • What is the difference between prototype and __proto__?

Additional Resources

Lesson Search Results

Showing top 10 results