TypeScript Modules
last modified March 3, 2025
Modules in TypeScript organize code into reusable, maintainable units. They
use import
and export
statements to share code
between files. This tutorial explores module syntax, default exports, and
namespaces with practical examples.
Basic Module Syntax
Modules allow exporting and importing variables, functions, and classes. This example demonstrates basic module usage.
export function add(a: number, b: number): number { return a + b; } export function subtract(a: number, b: number): number { return a - b; }
import { add, subtract } from './mathUtils'; console.log(add(10, 5)); // Output: 15 console.log(subtract(10, 5)); // Output: 5
The mathUtils
module exports two functions. The main
module imports and uses them.
Default Exports
A module can have one default export. Default exports simplify imports.
export default function log(message: string): void { console.log(`[LOG]: ${message}`); }
import log from './logger'; log("This is a log message."); // Output: [LOG]: This is a log message.
Default exports are imported without curly braces and can have any name.
Namespace Imports
Namespace imports group multiple exports under a single name.
export function add(a: number, b: number): number { return a + b; } export function subtract(a: number, b: number): number { return a - b; }
import * as MathUtils from './mathUtils'; console.log(MathUtils.add(10, 5)); // Output: 15 console.log(MathUtils.subtract(10, 5)); // Output: 5
Namespace imports are useful for organizing related functions.
Re-exporting Modules
Modules can re-export other modules to create a single entry point.
export function add(a: number, b: number): number { return a + b; } export function subtract(a: number, b: number): number { return a - b; }
export { add, subtract } from './mathUtils';
import { add, subtract } from './index'; console.log(add(10, 5)); // Output: 15 console.log(subtract(10, 5)); // Output: 5
Re-exporting simplifies imports by consolidating exports in one file.
Dynamic Imports
Dynamic imports load modules asynchronously at runtime.
export function add(a: number, b: number): number { return a + b; }
async function loadModule() { const mathUtils = await import('./mathUtils'); console.log(mathUtils.add(10, 5)); // Output: 15 } loadModule();
Dynamic imports are useful for code splitting and lazy loading.
Namespaces
Namespaces group related code and avoid global scope pollution.
namespace MathUtils { export function add(a: number, b: number): number { return a + b; } } export { MathUtils };
import { MathUtils } from './mathUtils'; console.log(MathUtils.add(10, 5)); // Output: 15
Namespaces are an alternative to modules for organizing code.
Best Practices
- Use Modules: Prefer modules over namespaces for modern apps
- Single Responsibility: Keep modules focused on one task
- Default Exports: Use default exports for single-purpose modules
- Dynamic Imports: Use dynamic imports for performance optimization
- Re-exporting: Consolidate exports in an
index.ts
file
Source
TypeScript Modules Documentation
This tutorial covered TypeScript modules with practical examples. Implement these patterns to write modular, maintainable code.
Author
List all TypeScript tutorials.