Prototypes are the mechanism by which JavaScript objects inherit features from one another.

The prototype chain

In the browser's console, try creating an object literal:

const myObject = {
    city: "Madrid",
    greet() {
        console.log(`Greetings from ${this.city}`);
    },
};

myObject.greet(); // Greetings from Madrid

This is an object with one data property, city, and one method, greet(). If you type the object's name followed by a period into the console, like myObject., then the console will pop up a list of all the properties available to this object. You'll see that as well as city and greet, there are lots of other properties!

city
greet
__defineGetter__
__defineSetter__
__lookupGetter__
__lookupSetter__
__proto__
constructor
hasOwnProperty
isPrototypeOf
propertyIsEnumerable
toLocaleString
toString
valueOf

What are these extra properties, and where do they come from?

Every object in JavaScript has a built-in property, which is called its prototype. The prototype is itself an object, so the prototype will have its own prototype, making what's called a prototype chain. The chain ends when we reach a prototype that has null for its own prototype.

When you try to access a property of an object: if the property can't be found in the object itself, the prototype is searched for the property. If the property still can't be found, then the prototype's prototype is searched, and so on until either the property is found, or the end of the chain is reached, in which case undefined is returned.

So when we call myObject.toString(), the browser:

  • looks for toString in myObject
  • can't find it there, so looks in the prototype object of myObject for toString
  • finds it there, and calls it

Shadowing properties

What happens if you define a property in an object, when a property with the same name is defined in the object's prototype? Let's see:

const myDate = new Date(1995, 11, 17);

console.log(myDate.getTime()); // 819144000000

myDate.getTime = function () {
    console.log("Something else!");
};

myDate.getTime(); // Something else!

This should be predictable, given the description of the prototype chain. When we call getTime() the browser first looks in myDate for a property with that name, and only checks the prototype if myDate does not define it. So when we add getTime() to myDate, then the version in myDate is called.

This is called "shadowing" the property.

Prototype Inheritance

Properties or methods added to the prototype of a constructor function are accessible to all objects derived from it. For example:

function Car(model, year) {
    this.model = model;
    this.year = year;
};

// Create multiple objects
let c1 = new Car("Mustang", 1964);
let c2 = new Car("Corolla", 1966);

// Add property
Car.prototype.color = "Red";

// Add method
Car.prototype.drive = function() {
    console.log(`Driving ${this.model}`);
};

// Display added property using c1 and c2 objects
console.log(`${c1.model} color: ${c1.color}`); // Mustang color: Red
console.log(`${c2.model} color: ${c2.color}`); // Corolla color: Red

// Display added method using c1 and c2 objects
c1.drive(); // Driving Mustang
c2.drive(); // Driving Corolla

In the above example, we created the objects c1 and c2 using the Car() constructor.

We then added the following to the prototype of Car():

- The color Property

  • It has a value of Red
  • Both c1 and c2 can access it using c1.color and c2.color, respectively

- The drive() Method

  • It is a method that displays a message
  • Both c1 and c2 can access it using c1.drive() and c2.drive(), respectively