Objects are a data type that can store multiple pieces of data as key:value pairs called properties
In other languages, they are referred to as dictionaries:
// Arrays store values in an order
const words = [
"hello",
"rainbow",
"cat"
]
// Objects/Dictionaries store key:value pairs
const dictionary = {
"hello": "a casual greeting",
"rainbow": "a colorful arc of light",
"cat": "the superior pet"
}
Objects are useful for storing collections of data related to a single "thing", like data about a user.
Objects can have arrays and other objects nested inside.
When creating an object, the quotes around the key name are optional, except when the key includes spaces or begins with a number:
const dictionary = {
hello: "a casual greeting",
rainbow: "a colorful arc of light",
cat: "the superior pet",
"see ya": "a casual way to say 'see you later'"
}
Accessing Objects with Dot Notation and Bracket Notation
Object values can be accessed and/or modified using dot notation or bracket notation
// Dot notation is the faster to type and easier to read than bracket notation
console.log(`Hi, my name is ${user.username}`);
// Bracket notation requires a string to be provided
console.log(`Hi, my name is ${user["username"]}`);
// Dot (and bracket) notation can be used to modify existing values, or create new ones!
user.password = 'try to break in now!';
user.isAdmin = false;
console.log(user);
// If a property holds an object or array, we can use dot/bracket notation on that value!
console.log(`My favorite meal is ${user.favoriteMeal.name}`);
console.log(`My best friend is ${user.friends[0]}`);
// Delete properties by using the `delete` keyword followed by dot/bracket notation
delete user.userId;
delete user["friends"];
Dynamic Properties Challenge
When the key name of a property is stored in a variable, we must use bracket notation (we can't use dot notation).
const key = 'some key value';
myObj[key] = 'newValue';
// Using dot notation here with the variable won't work.
// JS will think that "key" is the name of the key
myObj.key = 'newValue';
Complete the program below so that it lets users add words to the dictionary!
const prompt = require("prompt-sync")({ sigint: true });
const dictionary = {
"hello": "a casual greeting",
"rainbow": "a colorful arc of light",
"cat": "the superior pet"
}
while (true) {
console.log("Here are your words: ", dictionary);
const newWord = prompt("Add a word to your dictionary, or press q to exit. ");
if (newWord === 'q') { // a guard clause
break;
}
const definition = prompt(`Okay, what is the definition of ${newWord}? `);
// add the new word and its definition to the dictionary!
}
Solution
To complete this program, add the line dictionary[newWord] = definition; to the end
Iterating Over Objects
One of the key benefits of an Array is that we can easily iterate through its values with a for loop. This is possible because we can use the variable i to step through the indexes of the array.
const friends = ['bert', 'ernie', 'elmo'];
console.log("here are my friends:");
for (let i = 0; i < friends.length; i++) {
console.log(friends[i])
}
An object doesn't have countable indexes though. And their keys aren't in any particular order.
To iterate through an object, we can turn the object into an array using Object.keys(obj) which returns an Array of the given object's keys. We can then loop through those keys and use them to access their associated values:
const dictionary = {
"hello": "a casual greeting",
"rainbow": "a colorful arc of light",
"cat": "the superior pet"
}
// First get an array of the keys of the object
const words = Object.keys(dictionary);
console.log(words); // ["hello", "rainbow", "cat"]
// Then, iterate through them to get their values
for (let i = 0; i < words.length; i++) {
const word = words[i];
console.log(`The definition of ${word} is ${dictionary[word]}`);
}
If we don't care about the keys, we can just get an array of the object's values with Object.values(obj):
const dictionary = {
"hello": "a casual greeting",
"rainbow": "a colorful arc of light",
"cat": "the superior pet"
}
const definitions = Object.values(dictionary);
console.log("Can you tell what word each of these definitions are for?");
for (let i = 0; i < definitions.length; i++) {
console.log(definitions[i]);
}
Advanced Object Syntax
Object Shorthand Using Variables
When constructing an object from variables, it can be quite repetitive if the key name matches the variable name.
If we are storing the value held by a variable in a key with the same name as that variable, we can omit the : and just write the name of the variable:
This example makes a user object with the provided properties as well as some default values (isAdmin is false and an empty friends list).
This is the same syntax used when exporting "named exports":
const first = 'a';
const second = 'b';
const third = 'c';
module.exports = {
first,
second,
third
}
/*
This is shorthand for:
module.exports = {
"first": first,
"second": second,
"third": third,
}
*/
Destructuring
Destructuring is the process of creating variables from an existing array/object. When destructuring an object, the variable names must match the property key names that you wish to extract. The order in which you list the property names doesn't matter.
const dictionary = {
"hello": "a casual greeting",
"rainbow": "a colorful arc of light",
"cat": "the superior pet"
}
// Destructuring creates a variable for the properties of an object.
// Here, we are choosing to ignore rainbow.
const { hello, cat } = dictionary;
console.log(hello); // Prints "a casual greeting"
console.log(cat); // Prints "the superior pet"
This is the exact same syntax used when importing a "named import".
When an object is passed to a function, we will often destructure the object directly in the parameter list, letting us choose only the properties of the object that are relevant to the function:
const userBen = {
name: "Ben",
age: 28,
isAdmin: false
};
const introduceSelf = ({ name, age }) => {
console.log(`Hello! My name is ${name} and I am ${age} years old.`);
};
introduceSelf(userBen);