- TypeScript Tutorial
- TypeScript Home
- TypeScript Introduction
- TypeScript Setup
- TypeScript First Program
- TypeScript vs JavaScript
- TypeScript Data Types
- TypeScript Type Inference
- TypeScript Type Annotations
- TypeScript Interfaces
- TypeScript Enums
- TypeScript Type Aliases
- TypeScript Type Assertions
- TypeScript Variables
- TypeScript Functions
- TypeScript Functions
- TypeScript Optional Parameters
- TypeScript Default Parameters
- TypeScript Rest Parameters
- TypeScript Arrow Functions
- Classes and Objects
- Introduction to Classes
- Properties and Methods
- Access Modifiers
- Static Members
- Inheritance
- Abstract Classes
- Interfaces vs Classes
- Advanced Types
- TypeScript Union Types
- TypeScript Intersection Types
- TypeScript Literal Types
- TypeScript Nullable Types
- TypeScript Type Guards
- TypeScript Discriminated Unions
- TypeScript Index Signatures
- TypeScript Generics
- Introduction to Generics
- TypeScript Generic Functions
- TypeScript Generic Classes
- TypeScript Generic Constraints
- TypeScript Modules
- Introduction to Modules
- TypeScript Import and Export
- TypeScript Default Exports
- TypeScript Namespace
- Decorators
- Introduction to Decorators
- TypeScript Class Decorators
- TypeScript Method Decorators
- TypeScript Property Decorators
- TypeScript Parameter Decorators
- Configuration
- TypeScript tsconfig.json File
- TypeScript Compiler Options
- TypeScript Strict Mode
- TypeScript Watch Mode
Type Guards in TypeScript
Type guards are techniques in TypeScript that help refine the type of a variable within a specific scope. They allow TypeScript to infer more specific types for variables, improving type safety and reducing errors in your code. Type guards are typically used with conditional statements (if
, switch
) to narrow the type of a variable based on certain conditions.
Key Concepts of Type Guards
- Type Guard: A TypeScript feature that narrows the type of a variable within a block, allowing access to type-specific properties and methods.
- Custom Type Guards: You can define your own type guards using functions to check types dynamically.
- Built-in Type Guards: TypeScript provides built-in operators like
typeof
,instanceof
, andin
for performing type checks.
Example Usage of Type Guards
Example 1: Using typeof
for Primitive Types
TypeScript provides the typeof
operator to narrow types of primitive types like number
, string
, boolean
, etc.
function printLength(value: string | number): void {
if (typeof value === "string") {
console.log(`String length: ${value.length}`);
} else {
console.log(`Number value: ${value}`);
}
}
printLength("Hello"); // Output: String length: 5
printLength(123); // Output: Number value: 123
In this example:
- The
typeof
operator is used to check ifvalue
is astring
ornumber
. - Based on the result, the appropriate code is executed, refining the type within the respective block.
Example 2: Using instanceof
for Class Types
You can use instanceof
to narrow the type of an object based on its class or constructor function.
class Dog {
bark() {
console.log("Woof!");
}
}
class Cat {
meow() {
console.log("Meow!");
}
}
function makeSound(animal: Dog | Cat): void {
if (animal instanceof Dog) {
animal.bark(); // Dog specific method
} else {
animal.meow(); // Cat specific method
}
}
makeSound(new Dog()); // Output: Woof!
makeSound(new Cat()); // Output: Meow!
Here:
instanceof
is used to check whetheranimal
is an instance of theDog
orCat
class, narrowing down the type and ensuring correct method calls.
Example 3: Using in
Operator for Object Types
The in
operator checks whether a property exists in an object, which can be used to narrow down types in union types.
interface Car {
drive(): void;
}
interface Boat {
sail(): void;
}
function move(vehicle: Car | Boat): void {
if ("drive" in vehicle) {
vehicle.drive(); // Car specific method
} else {
vehicle.sail(); // Boat specific method
}
}
move({ drive: () => console.log("Driving") }); // Output: Driving
move({ sail: () => console.log("Sailing") }); // Output: Sailing
Here:
- The
in
operator is used to check if thedrive
property exists in thevehicle
, allowing the narrowing of type based on the object's structure.
Custom Type Guards
TypeScript allows you to create custom type guards, which are functions that return a type predicate (x is Type
) to narrow the type of a variable.
function isDog(animal: Dog | Cat): animal is Dog {
return (animal as Dog).bark !== undefined;
}
function makeSound(animal: Dog | Cat): void {
if (isDog(animal)) {
animal.bark(); // Dog specific method
} else {
animal.meow(); // Cat specific method
}
}
Here:
- The
isDog
function is a custom type guard that checks if theanimal
has thebark
method (indicating it is aDog
). - The
animal is Dog
syntax is the type predicate, which refines the type ofanimal
within theif
block.
Type Guards with Union Types
Union types often require type guards to narrow down the type so that specific properties can be accessed.
function getLength(value: string | number | null): number {
if (typeof value === "string") {
return value.length; // Accessing string-specific property
} else if (typeof value === "number") {
return value.toString().length; // Accessing number-specific property
} else {
return 0; // Handling null case
}
}
console.log(getLength("Hello")); // Output: 5
console.log(getLength(12345)); // Output: 5
console.log(getLength(null)); // Output: 0
Here:
- The
typeof
type guards ensure that the appropriate method (length
for strings ortoString()
for numbers) is called.
Summary
- Type Guards: TypeScript allows narrowing the types of variables using built-in operators (
typeof
,instanceof
,in
) or custom functions. - Built-in Type Guards: Use
typeof
for primitives,instanceof
for class instances, andin
for checking properties in objects. - Custom Type Guards: You can create your own type guards with functions returning a type predicate (
x is Type
). - Union Types: Type guards help to narrow the type in union types, providing type-safe access to properties.
Type guards provide powerful type safety by refining the types within specific blocks, enabling more precise and error-free code in TypeScript.