CommonJS – это стандартный формат модулей для языка JavaScript, который позволяет организовывать код в небольшие, независимые блоки, называемые модулями. Однако, важно понимать, что модульная система CommonJS не является частью самого языка, но она широко используется в среде Node.js для организации серверного кода.
Одними из ключевых понятий в CommonJS являются функции require, module.exports и exports. Функция require используется для загрузки модулей, то есть включения кода других модулей в текущий. Она принимает путь к модулю в качестве аргумента и возвращает экспортированный объект или функцию из этого модуля.
Для экспорта функций, объектов или переменных из модуля внешнему коду используются два объекта: module.exports и exports. Оба объекта являются частью встроенного модуля module и по сути являются одним объектом. Однако, module.exports используется тогда, когда нужно экспортировать значение, являющееся простым объектом или функцией, в то время как exports является ссылкой на module.exports и используется для экспорта нескольких значений или для дополнения module.exports, например, для добавления свойств или методов к уже существующему экспортированному объекту.
CommonJS и его роль в разработке на Node.js
Один из главных компонентов CommonJS - это функция require
. Она используется для загрузки модулей внутри других модулей. Функция require
принимает путь к модулю и возвращает экспортируемый объект, который может быть использован в текущем модуле.
Для экспорта функций, объектов или классов из модуля используются объекты module.exports
и exports
. Обычно, для экспорта используется module.exports
. Но, если вы хотите экспортировать только один объект, вы можете использовать поле exports
. Оба этих объекта являются ссылками на один и тот же объект по умолчанию, но при экспорте они могут быть присвоены различным значениям.
Структура модулей в CommonJS позволяет управлять зависимостями и организовывать код в логически связанные части. Каждый модуль может ссылаться на другие модули, используя функцию require
. Это позволяет легко подключать и использовать функционал, определенный в других модулях.
CommonJS предоставляет множество возможностей для разработки на Node.js. Она делает код более организованным, упрощает поддержку, и повышает переиспользуемость кода. Благодаря модульной системе CommonJS, разработчики могут создавать более масштабируемые и надежные приложения на платформе Node.js.
Импорт и экспорт модулей в CommonJS
В CommonJS есть два главных механизма для импорта и экспорта модулей: require
и module.exports
/exports
.
С помощью require
можно импортировать другие модули. Он принимает путь к модулю и возвращает объект, экспортированный из этого модуля.
Пример использования require
:
const module1 = require('./module1');
Данный пример импортирует модуль module1
из текущей папки.
С помощью module.exports
и exports
можно экспортировать объекты, функции или значения из модуля.
Пример экспорта:
module.exports = {
name: 'John',
age: 30,
greet: function() {
console.log('Hello!');
}
};
В этом примере мы экспортируем объект с полями name
, age
и методом greet
.
Комбинируя использование require
и module.exports
/exports
, можно создавать сложную структуру зависимостей и переиспользовать код по всему проекту.
Обратите внимание, что module.exports
является главным экспортом модуля, а exports
представляет собой ссылку на него. Весь экспорт должен быть осуществлен через module.exports
.
В итоге, благодаря механизмам импорта и экспорта, предоставляемым CommonJS, разработчики могут организовывать свой код в модули, легко переиспользовать его и создавать модульные приложения.
Различие между module.exports и exports
module.exports представляет собой ссылку на объект, который будет являться экспортируемым значением модуля. Этот объект может быть функцией, объектом, примитивным значением или даже классом. При присвоении значения module.exports, оно будет возвращено при импорте модуля.
На другой стороне, exports представляет собой ссылку на тот же объект, что и module.exports. Однако, exports - это не верхний уровень модуля, а свойство объекта module.exports. При использовании exports для экспорта значений, необходимо использовать синтаксис точки (например, exports.myFunction = ...), чтобы добавить свойства в module.exports.
Важно отметить, что при использовании exports без присваивания (например, exports = ...), ссылка на module.exports не изменяется, и изменения значения exports не будут иметь эффекта при импорте модуля.
Таким образом, основное различие между module.exports и exports заключается в том, что module.exports может быть присвоено напрямую, в то время как exports используется для добавления свойств в module.exports.
При экспорте модуля лучше всего использовать module.exports, если вы хотите экспортировать единственное значение, и exports, если вы хотите экспортировать несколько значений. Однако, в большинстве случаев module.exports будет предпочтительным способом экспорта.
Использование require в CommonJS для импорта модулей
Для использования require, вы должны сперва определить модуль, который вы хотите импортировать, при этом обычно указывается путь к файлу. Путь может быть относительным или абсолютным. Require также поддерживает некоторые расширения файлов по умолчанию, такие как .js и .json, поэтому вы не всегда должны указывать их.
После определения модуля с помощью require, вы можете присвоить его в переменную и использовать функции или переменные, экспортированные из этого модуля. Например:
- const myModule = require('./myModule');
- const myVariable = myModule.myVariable;
- myModule.myFunction();
При использовании require в CommonJS также важно помнить о кэшировании модулей. Если вы импортируете модуль один раз, он будет загружен в память и сохранен. При последующих импортах этот модуль будет браться из кэша, и файл не будет загружаться снова. Это может быть полезно, но также стоит быть осторожным при изменении модулей, так как они могут быть использованы в разных частях вашего проекта.
Таким образом, использование require в CommonJS предоставляет возможность импортировать модули из других файлов, делая ваш код более организованным, масштабируемым и читабельным.
Работа require с относительными и абсолютными путями
В CommonJS модули могут подключаться с использованием функции require. Она позволяет подключать другие модули в текущий файл.
Require может работать с относительными и абсолютными путями. Относительные пути указываются относительно текущего файла, а абсолютные пути начинаются с корневой папки проекта.
При использовании относительных путей, require сначала ищет модуль в текущей папке, а затем в подпапках. Если модуль не найден, происходит ошибка. Например, чтобы подключить модуль, находящийся в той же папке, что и текущий файл, можно использовать следующий синтаксис:
const module = require('./module');
Если модуль находится в подпапке, путь указывается относительно текущего файла и используются слэши для разделения папок:
const module = require('./subfolder/module');
При работе с абсолютными путями, require начинает поиск модуля с корневой папки проекта. Путь указывается полностью, начиная с корня. Например:
const module = require('/home/user/project/module');
Обратите внимание, что использование абсолютных путей может вызывать проблемы при переносе проекта на другую машину или при развертывании на сервере, поэтому рекомендуется использовать относительные пути, если это возможно.
Как работает module.exports в CommonJS
Чтобы экспортировать что-либо из модуля, вы можете присвоить значение module.exports. Например, вы можете экспортировать функцию следующим образом:
module.exports = function() {
console.log('Это экспортированная функция');
}
Или вы можете экспортировать объект:
module.exports = {
name: 'Объект',
method: function() {
console.log('Это метод объекта');
}
}
Если вы хотите экспортировать несколько функций или объектов, вы можете присвоить им значения внутри module.exports:
module.exports = {
fn1: function() {
console.log('Это первая функция');
},
fn2: function() {
console.log('Это вторая функция');
},
obj: {
prop: 'Это свойство объекта'
}
}
Когда вы импортируете модуль с помощью функции require, вы получаете доступ к экспортированным значениям, используя переменную, которой вы присвоили результат require. Вы можете использовать экспортированные функции, объекты или значения в своем коде:
const exportedFunction = require('./модуль');
const exportedObject = require('./модуль');
exportedFunction(); // "Это экспортированная функция"
console.log(exportedObject.name); // "Объект"
Также обратите внимание, что вы можете использовать сокращенную форму экспорта с помощью объекта exports, который является ссылкой на module.exports. Например:
exports.fn3 = function() {
console.log('Это третья функция');
}
Этот код эквивалентен следующему:
module.exports.fn3 = function() {
console.log('Это третья функция');
}
Однако, при использовании module.exports вместо exports можно исключить возможность случайного переопределения exports, поэтому рекомендуется всегда использовать module.exports для экспорта значений.
Как работает exports в CommonJS
В системе модулей CommonJS существует специальный объект exports, который позволяет экспортировать функции, объекты и переменные из модуля для использования в других модулях. Он представляет собой пустой объект по умолчанию, но может быть изменен для добавления свойств и методов.
Когда модуль загружается с помощью функции require, любые элементы, добавленные в объект exports, становятся доступными через импортирование модуля в другие части программы. Это делает exports ключевым механизмом для организации кода в модули и предоставления публичного интерфейса для использования во внешних модулях.
Аналогично объекту exports, существует альтернативный механизм экспорта – использование ключевого слова module.exports. В отличие от exports, module.exports может быть непосредственно присвоен объекту, функции или простому значению, переопределяя полностью объект exports. Однако следует обратить внимание, что переопределение exports приводит к потере ссылки на объект module.exports.
Важно помнить, что exports и module.exports являются ссылками на один объект в памяти, поэтому результаты не должны быть противоречивыми. Например, если в модуле экспортируется какое-то значение через exports, а потом module.exports переопределяется на новый объект, то модуль будет экспортировать только то, что указано в module.exports.
Как использовать require.resolve для нахождения пути к модулю
В CommonJS существует встроенная функция require.resolve
, которая позволяет найти полный путь к модулю без его загрузки.
Для использования require.resolve
необходимо передать ей имя модуля в качестве аргумента. Функция вернет абсолютный путь к файлу модуля, относительно корневой директории проекта.
Пример использования:
const path = require('path');
const modulePath = require.resolve('lodash');
console.log(modulePath); // /полный/путь/к/модулю/lodash.js
const relativeModulePath = path.relative(process.cwd(), modulePath);
console.log(relativeModulePath); // путь/к/модулю/lodash.js
Функция require.resolve
особенно полезна, если вам необходимо получить путь к модулю для последующей обработки или анализа его содержимого, но загружать модуль не требуется.
Кроме того, require.resolve
позволяет резолвить не только модули, но и относительные пути:
const path = require('path');
const modulePath = require.resolve('./utils');
console.log(modulePath); // /полный/путь/к/модулю/utils.js
const relativeModulePath = path.relative(process.cwd(), modulePath);
console.log(relativeModulePath); // путь/к/модулю/utils.js
Таким образом, использование require.resolve
позволяет получить полный путь к модулю в CommonJS, без его загрузки, что может быть полезно во множестве сценариев разработки и сборки JavaScript приложений.
Правила приоритета при импорте модулей в CommonJS
При работе с модулями в CommonJS есть несколько правил приоритета, которые определяют порядок импорта и экспорта модулей. Эти правила помогают разработчикам организовать код и предотвратить возможные проблемы и конфликты.
Основные правила приоритета в CommonJS:
Правило | Описание |
---|---|
1 | Приоритет импорта модулей, установленный в зависимостях |
2 | Приоритет экспорта модулей, установленный в коде модуля |
3 | Приоритет импорта модулей по умолчанию |
Приоритет импорта модулей, установленный в зависимостях, означает, что если модуль импортируется из другого модуля, то в первую очередь будет использован модуль, указанный в зависимостях.
Приоритет экспорта модулей, установленный в коде модуля, означает, что если модуль экспортирует несколько значений, то будет использовано только то значение, которое указано в коде модуля.
Приоритет импорта модулей по умолчанию означает, что если модуль не импортируется из другого модуля явно, то будет использован модуль по умолчанию.
Учитывая эти правила приоритета, разработчики могут более гибко управлять импортом и экспортом модулей в CommonJS, обеспечивая правильную работу и предотвращая потенциальные проблемы.
Каким образом CommonJS отличается от ES6 Modules
CommonJS создан для использования на стороне сервера и нацелен на синхронные операции. Он предлагает простую и интуитивно понятную схему импорта и экспорта модулей с использованием ключевых слов require и module.exports. Такой подход особенно удобен при работе с файловой системой или базой данных, где доступ к ресурсам осуществляется асинхронно.
ES6 Modules является стандартом ECMAScript 6 и предлагает новый синтаксис импорта и экспорта модулей. В отличие от CommonJS, ES6 Modules поддерживает асинхронную загрузку модулей и имеет более высокую производительность. Его синтаксис более декларативен и позволяет импортировать и экспортировать отдельные экспорты по имени.
Также стоит отметить, что CommonJS является синхронным, в то время как ES6 Modules поддерживает асинхронную загрузку модулей, что сделано для повышения производительности и улучшения опыта разработчика.
Использование CommonJS в браузере с помощью Browserify
Установить Browserify можно с помощью пакетного менеджера npm:
npm install -g browserify
После установки вы можете использовать команду browserify
для трансформации модулей CommonJS в браузерно-понятный код.
Например, у нас есть файл main.js
, который содержит следующий код:
var foo = require('./foo.js');
console.log(foo.bar());
Чтобы включить этот код в браузер, выполните следующую команду:
browserify main.js -o bundle.js
Теперь вы можете подключить файл bundle.js
к HTML-странице с помощью тега <script>
и использовать модуль main.js
в браузере:
<script src="bundle.js"></script>
Таким образом, благодаря Browserify мы можем использовать модули, основанные на CommonJS, в браузере, благодаря чему наш код становится более структурированным и модульным.