Notes Javascript - Objects
Objects
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 (or key) and a value. A property's value can be a function, in which case the property is known as a method.
Objects in JavaScript, just as in many other programming languages, can be compared to objects in real life. In JavaScript, an object is a standalone entity, with properties and type. Compare it with a cup, for example. A cup is an object, with properties. A cup has a color, a design, weight, a material it is made of, etc. The same way, JavaScript objects can have properties, which define their characteristics.
In addition to objects that are predefined in the browser, you can define your own objects.
Objects vs. JSON
A Javascript Object is a code object. Syntax-wise they look almost identical, but there is a subtle difference. JSON requires all identifiers to be wrapped in double quotes. JSON is meant to be used to store structured data and allow that structured data to be transmited just like XML and YAML. Objects are a programming construct. They are meant to be used in code.
A Simple Analogy
Imagine you have purchased furniture from a store, and you want it delivered. However the only one left in stock is the display model, but you agree to buy it.
In the shop, the chest-of-drawers you've purchased is a living object:
123456
However, you cannot send a chest-of-drawers in the post,
so you dismantle it (read, stringify it).
It is now useless in terms of furniture.
It is now JSON. It is in flat pack form.
Note: notice that the identifiers color and numberOfDrawers, in the JSON code below, are now wrapped in double quotes.
123
When you receive it, you then rebuild the chest-of-drawers (read, parse it). It is now back in object form.
Creating New Objects
You can create an object using an object initializer. Alternatively, you can first create a constructor function and then instantiate an object by invoking that function with the new operator.
Using Object Initializers
Object initializers are also called object literals. "Object initializer" is consistent with the terminology used by C++.
The syntax for an object using an object initializer is:
1234567
123456
1234567
1234567891011121314
1234567
1234567891011121314
12345678910111213141516171819
1234567891011
Object Property Names
An object property name can be any JavaScript string or symbol, including an empty string. However, you cannot use dot notation to access a property whose name is not a valid JavaScript identifier. For example, a property name that has a space or a hyphen, that starts with a number, or that is held inside a variable can only be accessed using the bracket notation. This notation is also very useful when property names are to be dynamically determined, i.e. not determinable until runtime. Examples are as follows:
12345678910111213141516171819202122232425
In the above code, the key anotherObj is an object, which is neither a string nor a symbol. When it is added to the myObj, JavaScript calls the toString() method of anotherObj, and use the resulting string as the new key.
You can also access properties with a string value stored in a variable. The variable must be passed in bracket notation. In the example above, the variable str held "myString" and it is "myString" that is the property name. Therefore, myObj.str will return as undefined.
123456789
This allows accessing any property as determined at runtime:
12345678910111213
Beware of Bracket Notation Injection Pitfalls
However, beware of using square brackets to access properties whose names are given by external input.
This may make your code susceptible to object injection attacks.
Nonexistent properties of an object have value undefined (and not null).
123
Enumerating Properties
There are three native ways to list/traverse object properties:
- for...in loops. This method traverses all of the enumerable string properties of an object as well as its prototype chain.
- Object.keys(myObj). This method returns an array with only the enumerable own string property names ("keys") in the object myObj, but not those in the prototype chain.
- Object.getOwnPropertyNames(myObj). This method returns an array containing all the own string property names in the object myObj, regardless of if they are enumerable or not.
You can use the bracket notation with for...in to iterate over all the enumerable properties of an object. To illustrate how this works, the following function displays the properties of the object when you pass the object and the object's name as arguments to the function:
12345678910111213
The term "own property" refers to the properties of the object, but excluding those of the prototype chain. So, the function call showProps(myCar, 'myCar') would print the following:
12345
The For-In code above is equivalent to:
123456789
Inheritance
All objects in JavaScript inherit from at least one other object. The object being inherited from is known as the prototype, and the inherited properties can be found in the prototype object of the constructor.
Defining Properties For All Objects Of One Type
You can add a property to all objects created through a certain constructor using the prototype property. This defines a property that is shared by all objects of the specified type, rather than by just one instance of the object. The following code adds a color property to all objects of type Car, and then reads the property's value from an instance car1.
1234
Defining Methods
A method is a function associated with an object, or, put differently, a method is a property of an object that is a function. Methods are defined the way normal functions are defined, except that they have to be assigned as the property of an object. See also method definitions for more details. An example is:
1234567891011121314
where objectName is an existing object, methodName is the name you are assigning to the method, and functionName is the name of the function.
You can then call the method in the context of the object as follows:
1
Methods are typically defined on the prototype object of the constructor, so that all objects of the same type share the same method. For example, you can define a function that formats and displays the properties of the previously-defined Car objects.
Method Type | Description |
---|---|
Constructor Method | Method code will be added to every instance of the object. |
Prototype Method | All instances of the object will share the method. |
123456
Notice the use of this to refer to the object to which the method belongs. Then you can call the displayCar method for each of the objects as follows:
1234
Using this For Object References
JavaScript has a special keyword, this, that you can use within a method to refer to the current object. For example, suppose you have 2 objects, Manager and Intern. Each object has its own name, age and job. In the function sayHi(), notice the use of this.name. When added to the 2 objects, the same function will print the message with the name of the respective object it's attached to.
1234567891011121314151617181920212223
this is a "hidden parameter" of a function call that's passed in by specifying the object before the function that was called. For example, in Manager.sayHi(), this is the Manager object, because Manager comes before the function sayHi(). If you access the same function from another object, this will change as well. If you use other methods to call the function, like Function.prototype.call() or Reflect.apply(), you can explicitly pass the value of this as an argument.
Defining Getters & Setters
A getter is a function associated with a property that gets the value of a specific property. A setter is a function associated with a property that sets the value of a specific property. Together, they can indirectly represent the value of a property.
Getters & Setters Can Be Either
- defined within object initializers, or
- added later to any existing object.
Within object initializers, getters and setters are defined like regular methods, but prefixed with the keywords get or set. The getter method must not expect a parameter, while the setter method expects exactly one parameter (the new value to set). For instance:
12345678910111213141516
The myObj object's properties are:
12345
Getters and setters can also be added to an object at any time after creation using the Object.defineProperties() method. This method's first parameter is the object on which you want to define the getter or setter. The second parameter is an object whose property names are the getter or setter names, and whose property values are objects for defining the getter or setter functions. Here's an example that defines the same getter and setter used in the previous example:
12345678910111213141516171819
Which of the two forms to choose depends on your programming style and task at hand. If you can change the definition of the original object, you will probably define getters and setters through the original initializer. This form is more compact and natural. However, if you need to add getters and setters later — maybe because you did not write the particular object — then the second form is the only possible form. The second form better represents the dynamic nature of JavaScript, but it can make the code hard to read and understand.
Returning Objects From Methods
12345678910111213141516171819202122232425