data:image/s3,"s3://crabby-images/44b4a/44b4a5a74b1b82a88041b38d7d2fb7e5ceae6fce" alt=""
Objects in TypeScript
Objects in TypeScript 관련
In TypeScript, objects are collections of properties where each property has a name (key) and a value. TypeScript allows us to define types for these properties, ensuring that objects conform to a specific structure.
let car = { car: 'Toyota', brand: 2024 };
console.log(car);
This works fine because TypeScript infers the types for car
and brand
automatically based on the values provided.
Explicit Object Types
When we want to define the shape of an object explicitly, we can use inline type annotations. This makes it clear what type each property should have. For example:
let carOne: { car: string; brand: number } = { car: 'Evil Spirit', brand: 2025 };
console.log(carOne);
This ensures that carOne
always has a car
property of type string
and a brand
property of type number
.
Let’s say we want to add a color
property to carOne
:
let carOne: { car: string; brand: number } = { car: 'Evil Spirit', brand: 2025, color: 'Black' };
The code above will show a redline because color
is not part of the defined type { car: string; brand: number }
.
data:image/s3,"s3://crabby-images/bfed3/bfed3e9396e4f81a0767fc369473b1c337de8593" alt="The error will look something like this"
Type '{ car: string; brand: number; color: string; }' is not assignable to type '{ car: string; brand: number; }'. Object literal may only specify known properties, and 'color' does not exist in type '{ car: string; brand: number; }'.
Similarly, if you try to change the type of brand
to a string
:
carOne.brand = "2026";
You’ll get another error:
Type 'string' is not assignable to type 'number'.
Having to write the full object type each time can get repetitive, especially for objects with many properties or when the same structure is used in multiple places. But don’t worry – I’ll soon introduce type aliases, which make defining and reusing object types much simpler. You’ll see how to use type aliases to simplify object types and make your code cleaner. After that, we’ll explore how to apply these concepts in React.
For now, focus on understanding the basics and how TypeScript enforces structure. It’s like peeking under the hood to see how TypeScript works behind the scenes.
Objects and Arrays
In TypeScript, we often deal with arrays of objects, where each object has a specific structure. TypeScript helps ensure that every object in the array conforms to the expected type.
Imagine you are managing a grocery store, and you want to keep track of your vegetables. Here’s how you might start:
let tomato = { name: 'Tomato', price: 2 };
let potato = { name: 'Potato', price: 1 };
let carrot = { name: 'Carrot' };
let vegetables: { name: string; price: number }[] = [tomato, potato, carrot];
When TypeScript checks this code, it throws an error because carrot
doesn’t have a price
property. The expected type for each item in the vegetables
array is { name: string; price: number }
. Since carrot
is missing the price
, TypeScript flags it as an error.
Type '{ name: string; }' is not assignable to type '{ name: string; price: number; }'. Property 'price' is missing in type '{ name: string; }' but required in type '{ name: string; price: number; }'.
If the price
is not always known or applicable (for example, maybe the carrot's price is still being negotiated), you can make the price
property optional. You can do this by adding a ?
after the property name:
let vegetables: { name: string; price?: number }[] = [tomato, potato, carrot];
Now, TypeScript knows that the price
property is optional. This means objects in the vegetables
array can either include price
or omit it without causing errors.
When a property is optional, TypeScript allows it to be either:
- Present with the specified type.
- Absent altogether.
This flexibility eliminates the error for objects like carrot
, which lack the price
property.
The** readonly
Modifi
In TypeScript, the readonly
modifier is a great way to ensure that certain properties or entire objects remain immutable. This is particularly useful when you want to prevent accidental changes to your data.
Let’s continue with our vegetable store example and see how readonly
works.
The Problem of Mutability
Imagine we have this setup:
let tomato = { name: 'Tomato', price: 2 };
let potato = { name: 'Potato', price: 1 };
let carrot = { name: 'Carrot' };
let vegetables: { name: string; price?: number }[] = [tomato, potato, carrot];
If someone accidentally tries to change the name
of the tomato
object or remove the carrot
object from the vegetables
array, TypeScript won’t complain:
vegetables[0].name = 'Cucumber'; // No error, but this could be unintended!
vegetables.pop(); // Removes the last vegetable, no warning.
We can use readonly
to make these objects and arrays immutable, ensuring their original state cannot be altered.
Readonly on Object Properties
To make the properties of each vegetable immutable, you can do the following:
let vegetables: { readonly name: string; readonly price?: number }[] = [
{ name: 'Tomato', price: 2 },
{ name: 'Potato', price: 1 },
{ name: 'Carrot' },
];
Now, if you try to change the name
or price
of any vegetable, TypeScript throws an error:
typescriptCopy codevegetables[0].name = 'Cucumber'; // Error: Cannot assign to 'name' because it is a read-only
Readonly Arrays
You can also make the entire vegetables
array immutable by declaring it as readonly
:
let vegetables: readonly { name: string; price?: number }[] = [
{ name: 'Tomato', price: 2 },
{ name: 'Potato', price: 1 },
{ name: 'Carrot' },
];
This prevents operations that modify the array itself, such as push
, pop
, or splice
:
vegetables.push({ name: 'Onion', price: 3 }); // Error: Property 'push' does not exist on type 'readonly { name: string; price?: number; }[]'.
vegetables.pop(); // Error: Property 'pop' does not exist on type 'readonly { name: string; price?: number; }[]'.
When to Use** `readonl
- Immutable data: Use
readonly
when you want to enforce immutability for objects or arrays, especially in contexts where data should remain constant (e.g., configurations, initial states, constants). - Prevent bugs: Protect your data from accidental changes caused by other parts of the code.
Complete Example
Here’s an updated example with readonly
in action:
let vegetables: readonly { readonly name: string; readonly price?: number }[] = [
{ name: 'Tomato', price: 2 },
{ name: 'Potato', price: 1 },
{ name: 'Carrot' },
];
// Attempting to modify data
vegetables[0].name = 'Cucumber'; // Error: Cannot assign to 'name' because it is a read-only property.
vegetables.pop(); // Error: Property 'pop' does not exist on type 'readonly { readonly name: string; readonly price?: number; }[]'.
console.log(vegetables);
Here’s what you should know about readonly, summarized:
readonly
on properties ensures individual fields of objects cannot be changed.readonly
on arrays makes the array itself immutable, preventing operations likepush
andpop
.- Combining both provides full immutability for objects within an array.
By using readonly
, you create safer, more predictable code, reducing bugs caused by unintended mutations.