30
ObjectOriented Programming in JavaScript Zander Magtipon May 25, 2015

Object Oriented Programming in JavaScript

  • Upload
    zand3rs

  • View
    86

  • Download
    2

Embed Size (px)

Citation preview

Object-­‐‑Oriented  Programming

 in  JavaScript

Zander Magtipon May 25, 2015

OOP  in  JavaScript• Overview • Prototype Chain • Custom Objects

o The class o The constructor o The object (class instance) o The property o The method o Static members o Private and privileged members

• Inheritance • Encapsulation • Accessing Superclass Members • Passing Constructor Arguments

Overview• JavaScript is designed on a simple object-based

paradigm. An object is a collection of properties, and a property is an association between a name and a value. A property's value can be a function, in which case the property is known as a method.

• One of the key differences of JavaScript from other OOP languages is that it does not have classes. Instead, JavaScript uses functions as classes.

• The class functionality is accomplished by object prototypes where object inherits from another object.

Overview• JavaScript functions are objects, giving functions the

capacity to hold executable code and be passed around like any other object.

• All objects in JavaScript are descended from Object object.

• All objects inherit methods and properties from Object.prototype.

• All object properties/methods are public.

Prototype  Chain• A prototype chain is a finite chain of objects which is used

to implement inheritance and shared properties.

• Every object in JavaScript has an internal link to another object called prototype. That prototype object has a prototype of its own, and so on until an object is reached with null as its prototype.

• __proto__ is the actual object that is used in the lookup chain to resolve methods, etc.

• prototype is the object that is used to build __proto__ when you create an object.

Prototype  Chainvar b = new Foo(20); var c = new Foo(30);

Custom  Objects

• Defining a class is as easy as defining a function.

function Person() { }

The class

Custom  Objects

• In JavaScript the function serves as the constructor of the object.

• The constructor is used to set the object's properties or to call methods to prepare the object for use.

function Person () {

console.log('instance created'); }

The constructor

Custom  Objects

• An instance of an object can be created by executing the constructor function using the new operator.

var person1 = new Person(); var person2 = new Person();

The object (class instance)

Custom  Objects

• Properties are set in the constructor of the class so that they are created on each instance.

• The keyword this, which refers to the current object, lets you work with properties from within the class.

The property (object attribute)

Custom  Objects

function Person(firstName) { this.firstName = firstName; console.log('Person instantiated'); }

var person1 = new Person('Alice'); var person2 = new Person('Bob');

// Show the firstName properties of the objects console.log('person1 is ' + person1.firstName); // logs "person1 is Alice" console.log('person2 is ' + person2.firstName); // logs "person2 is Bob"

The property (object attribute)

Custom  Objects

• Methods are functions that follow the same logic as properties. Calling a method is similar to accessing a property, but you add () at the end of the method name, possibly with arguments.

• To define a method, assign a function to a named property of the class's prototype property.

The method

Custom  Objects

function Person(firstName) { this.firstName = firstName; }

Person.prototype.sayHello = function() { console.log("Hello, I'm " + this.firstName); };

var person1 = new Person("Alice"); var person2 = new Person("Bob");

// call the Person sayHello method. person1.sayHello(); // logs "Hello, I'm Alice" person2.sayHello(); // logs "Hello, I'm Bob"

The method

Custom  Objects

• Static members (properties/methods) or class members only exist on the class and doesn't exist on child objects.

Static members

Custom  Objects

function Person(firstName) { this.firstName = firstName; }

Person.prototype.sayName = function() { console.log("instance:", this.firstName); };

Person.firstName = "anybody";

Person.sayName = function() { console.log("static:", this.firstName); };

var person1 = new Person("Alice"); person1.sayName(); // logs "instance: Alice" Person.sayName(); // logs "static: anybody"

Static members

Custom  Objects

• Private members are made by the constructor. Local vars and parameters of the constructor becomes the private members.

• A privileged method is able to access the private variables and methods, and is itself accessible to the public methods and the outside.

Private and privileged members

Custom  Objects

function Person(firstName) { //-- private var _firstName = firstName;

function _getMessage() { return "Hello my name is " + _firstName; } //-- privileged this.sayHello = function() { console.log(_getMessage()); } }

var person1 = new Person("Alice"); person1. sayHello(); // logs "Hello my name is Alice"

Private and privileged members

Inheritance• Inheritance is a way to create a class as a specialized

version of another class. • JavaScript only supports single inheritance. • When trying to access a property of an object, the

property will not only be sought on the object but on the prototype of the object, the prototype of the prototype, and so on until either a property with a matching name is found or the end of the prototype chain is reached.

• When an inherited function is executed, the value of this points to the inheriting object, not to the prototype object where the function is an own property.

Inheritancefunction Photo(name) { this.name = name || "photo"; }

Photo.prototype.upload = function() { console.log("Photo.upload:", this.name); };

ProfilePhoto.prototype = Object.create(Photo.prototype);

function ProfilePhoto(name) { Photo.call(this, name || "profile-photo"); }

var photo = new Photo(); photo.upload(); // logs "Photo.upload: photo"

var profilePhoto = new ProfilePhoto(); profilePhoto.upload(); // logs "Photo.upload: profile-photo"

Inheritancefunction Photo(name) { this.name = name || "photo"; }

Photo.prototype.upload = function() { console.log("Photo.upload:", this.name); };

ProfilePhoto.prototype = Object.create(Photo.prototype);

function ProfilePhoto(name) { Photo.call(this, name || "profile-photo"); }

//-- method override ProfilePhoto.prototype.upload = function() { console.log("ProfilePhoto.upload:", this.name); };

var photo = new Photo(); photo.upload(); // logs "Photo.upload: photo"

var profilePhoto = new ProfilePhoto(); profilePhoto.upload(); // logs "ProfilePhoto.upload: profile-photo"

Encapsulation• Encapsulation includes the idea that the data of an

object should not be directly exposed.

• Instead, callers that want to achieve a given result are coaxed into proper usage by invoking methods (rather than accessing the data directly).

Encapsulationfunction Photo(name) { this.name = name || "photo"; }

Photo.prototype.setName = function(name) { this.name = name; };

Photo.prototype.getName = function() { return this.name; };

var photo = new Photo(); photo.setName("picture"); // sets photo name to "picture" photo.getName(); // returns "picture"

Encapsulationfunction Photo(name) { var _name = name || "photo";

Object.defineProperty(this, "name", { get: function() { return _name; }, set: function(name) { _name = name; } }); }

var photo = new Photo(); photo.name = "picture"; // sets photo name to "picture" photo.name; // returns “picture”

Accessing  Superclass  Members

• One of the big differences between Classical (Object-Oriented) and Prototypal inheritance is that the former has an elegant mechanism for referring to the parent class (usually using the super keyword). It's often used in constructors to initialize the parent class with the supplied input parameters. Another common usage is to extend parent functionality in the child class.

Accessing  Superclass  Members

function Photo(name) { var _name = name || "photo";

Object.defineProperty(this, "name", { get: function() { return _name; } }); }

Photo.prototype.upload = function() { console.log("Photo.upload:", this.name); };

Accessing  Superclass  Members

ProfilePhoto.prototype = Object.create(Photo.prototype);

function ProfilePhoto(name) { Photo.call(this, name || "profile-photo"); }

//-- method override ProfilePhoto.prototype.upload = function() { Photo.prototype.upload.call(this); console.log(”ProfilePhoto.upload:", this.name); };

var profilePhoto = new ProfilePhoto(); profilePhoto.upload();

// logs "Photo.upload: profile-photo" // logs "ProfilePhoto.upload: profile-photo"

Accessing  Superclass  Members

ProfilePhoto.prototype = Object.create(Photo.prototype); function ProfilePhoto(name) { Object.defineProperty(this, "parent", { get: function() { return Photo; } }); this.parent.call(this, name || "profile-photo"); }

//-- method override ProfilePhoto.prototype.upload = function() { this.parent.prototype.upload.call(this); console.log(”ProfilePhoto.upload:", this.name); };

var profilePhoto = new ProfilePhoto(); profilePhoto.upload();

// logs "Photo.upload: profile-photo" // logs "ProfilePhoto.upload: profile-photo"

Passing  Constructor  Arguments

function Photo(name) { var _name = name || "photo";

Object.defineProperty(this, "name", { get: function() { return _name; } }); }

ProfilePhoto.prototype = Object.create(Photo.prototype);

function ProfilePhoto() { Photo.apply(this, Array.prototype.slice.call(arguments)); }

var profilePhoto = new ProfilePhoto("avatar"); // sets the name to "avatar"

Questions?

Thank  You!