Constructor Functions

Learning Goals

  • Create a constructor function
  • Use a constructor function to create an object
  • Explain what a constructor function is and how it works
  • Explain what new is and how it works with the constructor function

Introduction

JavaScript is sometimes a bit confusing. It has a native type Object. How does that relate to Object-Orientation? And how does the desire to avoid repeating ourselves relate to them both? In this lesson we'll work with plain old JavaScript Objects, and slowly transition to one of the key elements of OO, the constructor function and its partner, new.

Create a Constructor Function

It's sometime handy to represent data with objects, which gives us key/value pairs. For example, we may represent a user as the following:

const bobby = {name: 'bobby', age: 20, hometown: 'Philadelphia'}

Now imagine that we had a couple of users:

const bobby = {
  name: 'bobby',
  age: 20,
  hometown: 'Philadelphia'
}

const susan = {
  name: 'susan',
  age: 28,
  hometown: 'Boston'
}

Great. Two nice users.

Note, that with both objects sharing exactly the same keys, and only the values differing, we are repeating ourselves. We would like a mechanism to construct objects with the same attributes (that is, keys), while assigning different values to those keys. The name for what kind of function does this varies across many popular programming languages, but we'll call it a factory function because it spits out new instances.

function User(name, age, hometown) {
	return {
    name, // don't forget ES6 power-tools, this is the same as `name: name`
    age,
    hometown,
  }
}

let byronPoodle = User("Karbit's Byron By the Bay", 5, "Manhattan")
byronPoodle.age // => 5

Interestingly typeof confirms byronPoodle is an Object:

> typeof byronPoodle
'object'

However, something's not quite as clear as we might like it to be. If we ask byronPoodle what made it, the answer is...

> byronPoodle.constructor
[Function: Object]

byronPoodle is certainly an Object but it's more specific than that: it's a User. We'd really like for this special kind of object to be reflected when we ask it what it is. We'd like for a mystical process to come along and say you are not merely an Object, you are a User or a Dog. We tell JavaScript to bless the thing created by the constructor function into being something more specific than Object by using the keyword new. Using new requires that we evolve our factory function into a constructor function. It's the same idea, but with a few subtle additions.

Explain What new Is and How It Works With the Constructor Function

Lets create a constructor function. Constructor functions must be paired with the new keyword (which we'll cover in a moment).

function User(name, email) {
	this.name = name;
	this.email = email;
}

What's this here? It refers to the function's context. Since functions in JavaScript are also Objects, a function can say "on me, set a property." As we read this function we might think "OK, so you're going to run a function that will set properties on itself. That's not doing anything useful."

We would be right.

What we really want to say is something like "Hey, constructor function, when you run, create a new copy of yourself, leaving the original unchanged and on that particular copy, set the properties based on the arguments passed into the function. The keyword new tells the constructor function to do exactly that.

Put the two together like so:

function User(name, email) {
	this.name = name;
	this.email = email;
}

let lauren = new User("Lauren", "lauren@example.com");
lauren.name //=> "Lauren"

Remember you can and should try these out for yourself in the JavaScript console or in the node REPL

What's happening here hinges on the new keyword. When we invoke the function with new before it, you can imagine an imaginary JavaScript Object being copied for use in the User function. The constructor function is not changed; the freshly-created, new context is changed.

Here's the code sample above, but with some more comments. Follow the [numbers] 1-6.

function User(name, email) {
	this.name = name;   // [2] Set my context's property name to what
                      //     came in in the first argument (name)
	this.email = email; // [3] Set my context's property email to what
                      //     came in in the second argument (email)
}

// [1] Create a new "context", that's what `new` does
// Use _that_ new context inside of the execution of the `User` function
// also pass two parameters, "Lauren" and "lauren@example.com"

// [4]: Assign the new context thing with its this properties set to the
// variable `lauren`
let lauren = new User("Lauren", "lauren@example.com");

// [5]: Ask the new context for what's in its `.name` property
lauren.name //=> [6] "Lauren"

You can ask interesting questions about the lauren variable. Building on the previous code:

> typeof lauren
'object'
> lauren.constructor
[Function: User]

This sorta makes sense, the function that constructed lauren, or the constructor is User. The instance lauren is an object. Given what we know about the types available in JavaScript, object makes good sense (not a Number or a String, that's for sure!)

Since we know one OO pattern, we might be wondering how to add methods to the User instances. It should be obvious that if we can set a property to point to a value like "Lauren" or "lauren@example.com", we should be able to set an anonymous function to a property. That function would have access to the this context created when the instance was new'd into existence.

function User(name, email) {
	this.name = name;
	this.email = email;
	this.sayHello = function() {
		console.log(`Hello everybody, my name is ${this.name} whom you've been
mailing at ${this.email}!`);
	};
}

let lauren = new User('lauren', 'lauren@example.com');
lauren.sayHello(); //=> Hello everybody, my name is lauren whom you've been mailing at lauren@example.com!

Feel free to try it with other names and emails to prove to yourself that you can create instances from this constructor function.

Conclusion

In this lesson you've seen the constructor function and the new keyword work to create new instances with instance data and instance methods. We've pretty much got an object oriented language at our fingers! Not bad!

There's one problem with this design. It's incredibly memory inefficient. Take a look at the code and imagine what that inefficiency could be. We'll explore it together in the next lesson!