JavaScript Module Systems
JavaScript Module Systems
By Mohsen Mahabadi
10 min read
In this article, we are going to talk about different JS module systems, such as CommonJs, AMD, UMD, and ES Modules.
- Authors
- Name
- Mohsen Mahabadi
- linkedinMohsen Mahabadi
- Github
- githubmmahabadi
What is a JavaScript module?
In JavaScript, a module system allows developers to organize code into reusable pieces or components, which can be imported and used in other application parts. The module system provides a way to encapsulate code, manage dependencies, and control the scope of variables and functions.
Why do we need module systems?
Using a module system in JavaScript, or in any programming language, is crucial for several reasons, especially as applications grow in complexity. Here are some of the key reasons why a module system is essential:
1. Code Organization:
Modules allow developers to split their code into smaller, more manageable pieces, each responsible for a specific feature or functionality. This makes the codebase more organized, readable, and maintainable.
2. Scope Management and Collision Avoidance
Each module has its own scope, meaning variables and functions defined in a module are not globally accessible unless explicitly exported. This helps in avoiding variable conflicts and polluting the global namespace.
3. Dependency Management
Dependency management is clearer with modules, as dependencies are explicitly imported, making it easier to track and manage the relationships between different parts of an application.
Suppose you have three JavaScript files: utility.js
, dataProcessor.js
, and app.js
. The dataProcessor.js
file depends on a function from utility.js
, and app.js
depends on a function from dataProcessor.js
.
Available module systems in JavaScript
In JavaScript, there are several module systems that allow developers to organize and structure their code. Here are the most commonly used module systems:
- CommonJS (CJS)
- Asynchronous Module Definition (AMD)
- Universal Module Definition (UMD)
- ES6 Modules (ESM)
- SystemJS
1. CommonJS (CJS)
CommonJS is a module system implemented by Node.js for organizing and sharing JavaScript code. Before the advent of ES6 modules, CommonJS was the primary way to include and manage dependencies in Node.js applications. While browsers primarily use the ES6 module system, tools like Browserify and Webpack can transform CommonJS modules to be browser-compatible.
Key Concepts:
- Module Definition: In CommonJS, every file is its own module. The variables, functions, and objects you define in a file are local to that file unless explicitly exported.
- Export: To make parts of your code available for use in other files, you use
module.export
. - Import: To use a module in another file, you use the
require
function. This function returns whatever the target module exported. - Synchronous Loading: Modules are loaded synchronously, meaning the program waits for the module to be fully loaded and executed before moving on.
- Module Caching: Modules are cached after the first time they are loaded, improving performance and ensuring that module initialization only happens once.
- No tree shaking: because when you import a module you get an object.
How to use CommonJS?
Setting up:
Ensure you have Node.js installed. If not, download and install it from Node.js official website.
Step 1:
Create a file named counter.js
:
let count = 0
function increment() {
count++
}
function getCount() {
return count
}
module.exports = {
increment,
getCount,
}
Step2:
Create a file named app.js
:
const counter = require('./counter.js')
console.log(counter.getCount()) // Outputs: 0
counter.increment()
console.log(counter.getCount()) // Outputs: 1
Running the Code:
In your terminal, run the file using Node.js node app.js
. You’ll see that the count
variable in the counter.js
module retains its state between function calls.
2. Asynchronous Module Definition (AMD)
It is a specification for the modular development of JavaScript applications and libraries. It allows developers to define modules and their dependencies asynchronously, making it particularly well-suited for the browser environment where synchronous loading of modules can result in performance issues. The most popular implementation of AMD is the RequireJS library.
Key Concepts:
- Define a Module: Use the
define
function to specify a module, its dependencies, and a factory function that returns the module's exports. - Load and Use a Module: Use the
require
function to load a module and its dependencies. - Asynchronous Loading: One of the primary benefits of AMD is its support for asynchronous module loading. This means that modules and their dependencies can be fetched in parallel, which can lead to faster page loads in a browser environment.
- Module ID: Each module can have a unique module ID, which is typically its file path (minus the
.js
extension). This ID is used when specifying dependencies.
How to use AMD?
Setting up:
Include RequireJS in your HTML file:
<script data-main="scripts/app" src="path/to/require.js"></script>
The data-main
attribute points to the main script of your application.
Step 1:
Create a file named scripts/math.js
:
// math.js
define([], function () {
function add(a, b) {
return a + b
}
function subtract(a, b) {
return a - b
}
return {
add: add,
subtract: subtract,
}
})
Here, we’ve defined a module without any dependencies (empty array) and exported two functions.
Step 2:
Create a file named scripts/app.js
(your main script):
// app.js
require(['math'], function (math) {
console.log(math.add(5, 3)) // Outputs: 8
console.log(math.subtract(5, 3)) // Outputs: 2
})
Here, we’re loading the math
module and using its functions. Once the math
module is loaded, the function will be called.
Running the Code:
Simply open the HTML file containing the RequireJS script tag in a browser. The main script (app.js
) will be executed, and it will asynchronously load the math.js
module.
3. Universal Module Definition (UMD)
It is a pattern for writing JavaScript modules that can work both in the browser and on the server. The idea behind UMD is to make a module compatible with the most popular script loaders of the day. In fact, it is a combination of CommonJs + AMD.
Key Concepts:
Compatibility: UMD aims to support the three most popular module formats:
AMD (Asynchronous Module Definition): Used by browsers and includes libraries like RequireJS.
CommonJS (CJS): Used primarily on the server side, especially with Node.js.
Global variable definition: For browsers without any module loader.Flexibility: UMD allows developers to write a module once and have it work in many environments without modification.
Wrapper Pattern: UMD typically wraps the module definition inside a function that can detect the module system in use and act accordingly.
How to use UMD?
Here’s a basic example of how you might define a module using UMD:
1. Define a UMD module:
Create a file named my-umd.js
:
;(function (root, factory) {
if (typeof define === 'function' && define.amd) {
// AMD. Register as an anonymous module.
define([], factory)
} else if (typeof module === 'object' && module.exports) {
// Node. Does not work with strict CommonJS, but
// only CommonJS-like environments that support module.exports,
// like Node.
module.exports = factory()
} else {
// Browser globals (root is window)
root.myModule = factory()
}
})(this, function () {
return {
hello: function () {
return 'Hello, World!'
},
}
})
2. Running the UMD module:
a. In the Browser:
Include the UMD module in a script tag:
<script src="my-umd.js"></script>
<script>
console.log(myModule.hello()) // Outputs: "Hello, World!"
</script>
b. Using AMD (e.g., RequireJS):
First, set up RequireJS and then define the module:
<script src="path_to_require.js"></script>
<script>
require(['my-umd'], function (myModule) {
console.log(myModule.hello()) // Outputs: "Hello, World!"
})
</script>
c. In Node.js:
First, set up Node.js and then just require the file:
const myModule = require('./my-umd.js')
console.log(myModule.hello()) // Outputs: "Hello, World!"
4. ES6 Modules (ESM)
It is the native module system introduced in ECMAScript 6 (ES6), also known as ECMAScript 2015 (ES2015). ESM allows you to modularize your JavaScript code, making it more maintainable, scalable, and organized for both the browser and the server. With its native support in modern environments and tools, it’s become an essential part of the JavaScript ecosystem.
Key Concepts:
- Export & Import: The core of ESM is the ability to export and import functions, objects, or primitives from one module to another.
- Static Analysis: Imports and exports are determined statically at compile time, rather than runtime. This means tools can analyze a module’s dependencies without executing it.
- File-based Modules: In ESM, each file is its own module. Anything not exported remains private to that module.
- Dynamic Imports: Introduced later, dynamic imports allow you to load modules on the fly using promises.
- Static Structure: One of the defining characteristics of ESM is its static structure. This means you can’t conditionally import/export inside functions or conditionals. The structure is determined at compile time, not runtime.
- Cyclic Dependencies: ESM handles cyclic (or circular) dependencies between modules. If module A imports from module B, and module B imports from module A, the language has mechanisms to handle this without causing errors.
- Module Caching: If multiple modules import from a single module, that module will only be executed once. This ensures that side effects in a module (like setting up an event listener) don’t occur multiple times.
- Tree shakeable: because of
static analysis
supported by ES6.
ES6 Modules Syntax
Exporting and importing a module:
// math.js
export const PI = 3.14
export const power = (a, b) => a ** b
// Importing into another module:
// app.js
import { PI, power } from './math.js'
console.log(power(2, 3)) // Outputs: 8
Using Default Exports:
Each module can have one default export, which can be imported without curly braces.
// greet.js
export default function () {
return 'Hello!'
}
//or
const greeting = () => {
return 'Hello!'
}
export default greeting
// app.js
import greet from './greet.js'
console.log(greet()) // Outputs: Hello!
Combining Imports:
// app.js
import greet, { add, subtract } from './module.js'
Renaming Imports:
// app.js
import { add as addition } from './math.js'
console.log(addition(2, 3)) // Outputs: 5
Dynamic Imports:
Useful for code-splitting
and lazy-loading
modules.
const loadModule = async () => {
const module = await import('./dynamicModule.js')
module.someFunction()
}
How to use ESM?
Step 1:
Let's say you have a math.js
file structured as follows:
// math.js
export const add = (a, b) => a + b
export const subtract = (a, b) => a - b
Step 2:
You then utilize these functions in your index.js
:
import { add, subtract } from 'math'
console.log(add(2, 3)) // Outputs: 5
Running the Code:
To execute this in a browser, you'll need to include the index.js
script in your html file. It's crucial to specify type="module"
within the script tag to inform the browser to treat our file as a module. Additionally, the importmap
is necessary to facilitate the direct loading of JavaScript modules in the browser.
<script type="importmap">
{
"imports": {
"math": "./math.js"
}
}
</script>
<script type="module" src="./index.js"></script>
5. SystemJS
SystemJS is a dynamic module loader that provides a way to load ES6 modules, CommonJS, AMD, and global scripts. It’s a powerful tool that bridges the gap between different module formats and allows developers to write code that can be loaded in various environments without modification.
Where do we use SystemJs?
SystemJS is primarily designed for the browser to handle module loading in environments that may not natively support certain module formats. However, it's not limited to just the browser. You can use SystemJS on the server side, particularly with Node.js, but there are some considerations to keep in mind:
Native Module Support in Node.js: Node.js has built-in support for CommonJS modules, and recent versions of Node.js also support ES6 modules natively. This often reduces the need for a module loader like SystemJS on the server side.
Use Cases on the Server: One potential use case for SystemJS on the server is when you have a mixed module environment (e.g., a combination of CommonJS, AMD, and ES6 modules) and you want a unified way to load them. Another scenario might be dynamically loading modules at runtime based on certain conditions.
Integration with Other Tools: If you’re considering using SystemJS on the server side, you might also want to look into how it integrates with other server-side tools and frameworks you’re using.
Key Concepts:
- Universal Module Loading: can load multiple module formats, including: — ES6 modules
— CommonJS
— AMD
— UMD
— Global scripts (non-modular scripts) - Dynamic Loading: allows for the dynamic loading of modules at runtime. This is particularly useful for applications that load modules based on certain conditions or user interactions.
- Transpilation Support: can be integrated with transpilers like Babel or TypeScript. This means you can write code in modern ES6 or TypeScript and have SystemJS load and transpile it on the fly in browsers that only support ES5.
- Plugin System: supports plugins, enabling it to load not just JavaScript modules but also other resources like JSON, CSS, text, and images as modules.
- Configuration and Flexibility: provides a rich configuration system, allowing developers to specify paths, aliases, default extensions, and other settings to control how modules are loaded and resolved.
- Production Optimizations: Because of its dynamic loading features, SystemJS thrives in development contexts, but it also gives options to optimize for production. Tools like
systemjs-builder
can be used to bundle modules into fewer files, reducing the number of HTTP requests and improving load times. - Compatibility: provides a way to use modules in environments that don't natively support them. For instance, before ES6 module support was widespread in browsers, SystemJS allowed developers to use ES6 module syntax and load them effectively.
- Production Bundles: While SystemJS is great for development with its dynamic loading capabilities, for production, it's often recommended to create bundles to reduce the number of requests and improve performance.
How to use it?
Installation:
You can include SystemJS via a script tag or install it via npm npm install systemjs
.
<script src="path-to-systemjs/dist/system.js"></script>
Configuration:
Set up a configuration for SystemJS. This tells SystemJS where to find modules and how to load them.
SystemJS.config({
map: {
jquery: 'path-to-jquery',
},
packages: {
app: {
defaultExtension: 'js',
},
},
})
Loading Modules:
Use SystemJS to load your main module.
SystemJS.import('app/main.js').then(function (module) {
// Use the module here
})
Conclusion
JavaScript has seen various module systems over the years, each with its unique strengths. However, ESM Modules stand out due to their efficiency and widespread adoption in modern frameworks. For developers aiming for streamlined and future-ready applications, using ESM Modules is a clear choice.
Upcoming events
Drupal CMS Launch Party
Zoals sommigen misschien weten wordt op 15 Januari een nieuwe distributie van Drupal gelanceerd. Namelijk Drupal CMS (ook wel bekend als Starshot). Om dit te vieren gaan we op onze campus een klein eventje organiseren. We gaan die dag samen de livestream volgen waarbij het product gelanceerd wordt. De agenda is als volgt: 17u – 18u30: Drupal CMS livestream met taart 18u30 – 19u00: Versteld staan van de functionaliteiten 19u – 20u: Pizza eten en verder versteld staan van de functionaliteiten Laat ons zeker weten of je komt of niet door de invite te accepteren! Tot dan!
| Coven of Wisdom Herentals
Go to page for Drupal CMS Launch PartyCoven of Wisdom - Herentals - Winter `24 edition
Worstelen jij en je team met het bouwen van schaalbare digitale ecosystemen of zit je vast in een props hell met React of in een ander framework? Kom naar onze meetup waar ervaren sprekers hun inzichten en ervaringen delen over het bouwen van robuuste en flexibele applicaties. Schrijf je in voor een avond vol kennis, heerlijk eten en een mix van creativiteit en technologie! 🚀 18:00 – 🚪 Deuren open 18:15 – 🍕 Food & drinks 19:00 – 📢 Building a Mature Digital Ecosystem - Maarten Heip 20:00 – 🍹 Kleine pauze 20:15 – 📢 Compound Components: A Better Way to Build React Components - Sead Memic 21:00 – 🙋♀️ Drinks 22:00 – 🍻 Tot de volgende keer? Tijdens deze meetup gaan we dieper in op het bouwen van digitale ecosystemen en het creëren van herbruikbare React componenten. Maarten deelt zijn expertise over het ontwikkelen van een volwassen digitale infrastructuur, terwijl Sead je laat zien hoe je 'From Props Hell to Component Heaven' kunt gaan door het gebruik van Compound Components. Ze delen praktische inzichten die je direct kunt toepassen in je eigen projecten. 📍 Waar? Je vindt ons bij iO Herentals - Zavelheide 15, Herentals. Volg bij aankomst de borden 'meetup' vanaf de receptie. 🎫 Schrijf je in! De plaatsen zijn beperkt, dus RSVP is noodzakelijk. Dit helpt ons ook om de juiste hoeveelheid eten en drinken te voorzien - we willen natuurlijk niet dat iemand met een lege maag naar huis gaat! 😋 Over iO Wij zijn iO: een groeiend team van experts die end-to-end-diensten aanbieden voor communicatie en digitale transformatie. We denken groot en werken lokaal. Aan strategie, creatie, content, marketing en technologie. In nauwe samenwerking met onze klanten om hun merken te versterken, hun digitale systemen te verbeteren en hun toekomstbestendige groei veilig te stellen. We helpen klanten niet alleen hun zakelijke doelen te bereiken. Samen verkennen en benutten we de eindeloze mogelijkheden die markten in constante verandering bieden. De springplank voor die visie is talent. Onze campus is onze broedplaats voor innovatie, die een omgeving creëert die talent de ruimte en stimulans geeft die het nodig heeft om te ontkiemen, te ontwikkelen en te floreren. Want werken aan de infinite opportunities van morgen, dat doen we vandaag.
| Coven of Wisdom Herentals
Go to page for Coven of Wisdom - Herentals - Winter `24 editionThe Test Automation Meetup
PLEASE RSVP SO THAT WE KNOW HOW MUCH FOOD WE WILL NEED Test automation is a cornerstone of effective software development. It's about creating robust, predictable test suites that enhance quality and reliability. By diving into automation, you're architecting systems that ensure consistency and catch issues early. This expertise not only improves the development process but also broadens your skillset, making you a more versatile team member. Whether you're a developer looking to enhance your testing skills or a QA professional aiming to dive deeper into automation, RSVP for an evening of learning, delicious food, and the fusion of coding and quality assurance! 🚀🚀 18:00 – 🚪 Doors open to the public 18:15 – 🍕 Let’s eat 19:00 – 📢 First round of Talks 19:45 – 🍹 Small break 20:00 – 📢 Second round of Talks 20:45 – 🍻 Drinks 21:00 – 🙋♀️ See you next time? First Round of Talks: The Power of Cross-browser Component Testing - Clarke Verdel, SR. Front-end Developer at iO How can you use Component Testing to ensure consistency cross-browser? Overcoming challenges in Visual Regression Testing - Sander van Surksum, Pagespeed | Web Performance Consultant and Sannie Kwakman, Freelance Full-stack Developer How can you overcome the challenges when setting up Visual Regression Testing? Second Round of Talks: Omg who wrote this **** code!? - Erwin Heitzman, SR. Test Automation Engineer at Rabobank How can tests help you and your team? Beyond the Unit Test - Christian Würthner, SR. Android Developer at iO How can you do advanced automated testing for, for instance, biometrics? RSVP now to secure your spot, and let's explore the fascinating world of test automation together!
| Coven of Wisdom - Amsterdam
Go to page for The Test Automation Meetup