Как передать переменную из одного js файла в другой
В одном из моих JS файлов я включаю еще один. Как установить переменные во включенном модуле?
Я думал, что что-то вроде этого будет работать
И затем в mymodule
Но это не работает, оно undefined . Каковы различные варианты передачи значения в модуль? Я мог бы просто добавить переменную в качестве параметра для каждой функции, которую я вызываю в mymodule, но это не идеально.
Есть ли способ сделать это без globals, так что я могу самостоятельно устанавливать переменные в разных требуемых модулях?
NodeJS require() всегда будет загружать модуль один раз, поэтому вам нужно будет реализовать область видимости в вашем модуле, где могут существовать разные экземпляры модуля со своим собственным внутренним состоянием.
Вы можете реализовать свой модуль как класс JS, например:
Затем в вашем коде
Проблема с тем, что вы делали, заключается в том, что вы задали переменную после импорта, но this.myvariable === 'test'; вызывается, когда модуль был импортирован, прежде чем ваша переменная была установлена.
Вы можете настроить модуль на функцию, а затем вызвать функцию при импорте, передав переменную в качестве аргумента.
Если вы используете этот метод, имейте в виду, что вы получаете другой экземпляр своего модуля, куда бы вы ни импортировали, что может быть не таким, каким вы хотите.
Если вы хотите установить значения модуля извне модуля, хорошим вариантом является то, чтобы ваш модуль экспортировал объект с помощью метода setter и использовал его для установки значения переменной как свойства объекта. Это делает более понятным, что вы хотите, чтобы это значение было настраиваемым, тогда как просто выполнение myModule.myVar = может привести вас к путанице позже.
В этом случае вы получаете доступ к тому же экземпляру модели, где бы вы ее не импортировали.
Изменить в ответ на комментарий
В первом варианте вы показываете, где вы получаете другой экземпляр каждый время, как я могу экспортировать несколько функций, каждый из которых имеет один и тот же MYVAR? Если этот модуль экспортирует 5 функций, каждый из которых нуждается в myVar, могу ли я установите его в одном месте, как во время импорта, а не передайте его в каждая функция?
Не совсем уверен, понимаю ли я то, что вы описываете, но вы можете сделать что-то вроде этого:
Каждый из этих подмодулей будет иметь доступ к одному и тому же myVar. Таким образом, вы должны импортировать как указано выше, и результатом будет объект, содержащий каждый из ваших пяти модулей в качестве свойств. Я не могу сказать, хорошая ли это идея, она становится довольно запутанной, но, возможно, это имеет смысл для вашей ситуации.
Должно работать нормально. Вот рабочий пример:
Имейте в виду, если вы используете this в закрытии, что область действия больше не является самим модулем. Так что это может быть вашей проблемой.
Можете ли вы показать мне часть кода модуля, в которой вы используете это?
С первых дней существования Интернета веб-сайты разрабатывались в основном на HTML и CSS. И если JavaScript код все-таки загружался на страницу то, это обычно происходило в виде небольших фрагментов, которые обеспечивали простейшие визуальные эффекты или попытки сделать страничку более интерактивной. Поэтому программы на JavaScript чаще в объеме одного файла помещались в тег script . Разработчик мог также разбить код JavaScript на несколько скриптов (файлов), но даже при таком подходе все переменные и функции находящиеся в них будут добавлены в общую глобальную область видимости.
В настоящее время, при разработке современных приложений на Javascript, существует необходимость в использовании стороннего кода для решения общих задач, разделения кода на файлы отдельных модулей, а также внедрения комплекса мер для защиты от засорения глобального пространства имен.
Спецификация ECMAScript 2015 представила новое средство языка JavaScript: модули. Которые позволяют использовать ключевые слова import и export , уже наверняка знакомые по другим языкам программирования. Из этого руководства вы узнаете, что такое модуль JavaScript, и как использовать инструкции import и export для правильной организации кода в соответствии с требованиями спецификации.
Модульное программирование
До появления концепции модулей в JavaScript, в случае если разработчик хотел организовать свой код как отдельные сегменты кода, он создавал несколько файлов и связывал их как отдельные сценарии. Чтобы продемонстрировать этот подход, создадим для примера следующие файлы: index.html и два файла JavaScript, functions.js и script.js .
В файл index.html будет отображаться сумма, разность, произведение и частное двух чисел, в него же в тегах script поместим ссылку на два JavaScript файла.
И так создайте файл index.html в текстовом редакторе и добавьте в него следующий код:
Этот HTML код будет отображать в h2 заголовке страницы значения переменных x и y , а также результат операций с этими переменными в следующих за ним элементах p . Значения атрибутов id для элементов устанавливаются путем манипулирования DOM , которое будет производиться в коде script.js . Этот код также будет устанавливать начальные значения переменных x и y .
Файл functions.js содержит математические функции, которые используются в первом сценарии script.js .
Откройте файл functions.js и добавьте в него следующий код:
И наконец, код в файле script.js определяет значения переменных x и y , применяет к их значениям математические функции и отображает результат:
После подключения к index.html Javascript файлов, можете открыть его в браузере, где увидите следующий результат.
Для веб-сайтов в которых используются несколько небольших скриптов это достаточно эффективный способ: просто разделить код по разным скриптам (файлам). Однако с этим подходом связаны некоторые проблемы, в том числе:
- Загрязняется глобальное пространство имен. Все переменные, которые вы создали в своих сценариях: sum , difference и т.д., теперь находятся в глобальном объекте window . И если вы попытаетесь использовать другую переменную с именем sum , вызываемую в другом файле, то будет трудно разобраться, какое значение будет использоваться в текущем сценарии, поскольку все они будут использовать одну и ту же глобальную переменную window.sum . Единственный способ сделать переменную приватной private: это поместить ее в область действия некоторой функции. Однако при этом может возникнуть конфликт, например, между значением x атрибута id элемента DOM и переменной var x .
- Усложняется управление зависимостями. Сценарии должны загружаться в заданном порядке следования сверху вниз, чтобы корректно обеспечить доступность всех переменных в нужное время. Сохранение сценариев в виде разных файлов создает иллюзию разделения, но по сути это то же самое, что сохранение всего Javascript кода вместе в inline теге <script> .
Прежде чем ES6 добавил собственные модули к языку JavaScript, сообщество попыталось предложить несколько решений. Первые решения были написаны на ванильном JavaScript, например, весь код записывался в объект или немедленно вызываемое функциональное выражение (IIFE), а затем помещался в один объект в глобальном пространстве имен. Это было лучшим решением, чем подход использования нескольких файлов скриптов, но по-прежнему возникали те же проблемы с наличием одного объекта в глобальное пространстве имен и это не решало проблемы эффективного разделения кода сторонних библиотек.
После этого появилось несколько других решений, реализующих модульность Javascript кода: CommonJS , синхронный подход, реализованный в Node.js, Asynchronous Module Definition (AMD), который реализовывал асинхронный подход, а также Universal Module Definition (UMD) , который задумывался как универсальный подход, поддерживающий оба предыдущих стиля разделения кода.
Появление этих решений упростило разработчикам совместное и повторное использование кода в форме пакетов, аналогов модулей, которые можно распространять и совместно использовать, например, их можно найти на npm . Несмотря на то, что было разработано много решений, ни одно из них не было непосредственно встроено в язык JavaScript, и для использования модулей в браузерах пришлось реализовать такие инструменты, как Babel , Webpack или Browserify .
Из-за множества проблем, связанных с многофайловым подходом и сложностью предложенных решений, разработчики были заинтересованы в переносе модульного подхода к программированию в язык JavaScript. По этой причине стандарт ECMAScript 2015 теперь поддерживает использование модулей JavaScript.
Модуль представляет собой сборку кода bundle и выступает в качестве интерфейса для обеспечения функциональных возможностей использования отдельных модулей. Модуль экспортирует свой код и импортируется для использования в другом коде. Модули полезны тем, что позволяют разработчикам повторно использовать код, при этом обеспечивают стабильный код и понятный интерфейс к нему, который могут использовать другие разработчики, при этом их использование не загрязняет глобальное пространство имен.
Модули (называемые также модулями ECMAScript или модулями ES) теперь доступны нативно (встроены в JavaScript), и в оставшейся части этого руководства вы узнаете, как использовать их в своем коде.
Модули в JavaScript
Ключевые слова import и export используются для работы с модулями в JavaScript:
- import : применяется для чтения кода, экспортируемого из другого модуля.
- export : применяется для предоставления кода другим модулям.
Чтобы продемонстрировать, как это можно использовать на практике, обновим файл functions.js до модуля и экспортируем его функции. Для этого просто добавим ключевое слово export перед каждой функцией, что автоматически сделает их код доступным для любого другого модуля.
Добавим в файл следующий код:
Теперь в верхней части файла script.js , нам необходимо использовать ключевое слово import для извлечения (импорта) кода функций из модуля functions.js .
Примечание : инструкции кода с ключевым словом import всегда должны быть вверху файла перед любым другим кодом, при этом необходимо указать относительный ( ./ в нашем случае случае), либо абсолютный путь к импортируемому файлу модуля.
Добавьте следующий код в файл script.js :
Обратите внимание, что отдельные функции, импортируются из модуля путем помещения их имени в фигурные скобки.
Чтобы этот код загружался в браузере как модуль, а не как обычный скрипт, добавьте в файл index.html к тегу script атрибут type="module" . Любой код, который использует инструкции import или export должен содержать указанный атрибут при подключении соответствующего файла модуля:
Теперь вы сможете перезагрузить обновленную страницу, и ваш веб-сайт будет использовать нативный механизм модулей Javascript. Поддержка браузерами этой новой возможности достаточно высока, используйте сервис caniuse, чтобы узнать какие браузеры ее поддерживают. Обратите внимание, что если вы загружаете файл модуля путем помещения в значение атрибута src прямой ссылки на локальный файл, то столкнетесь со следующей ошибкой:
Модули отличаются от обычных скриптов несколькими существенными особенностями:
- Модули ничего не добавляют в global глобальную область видимости (или в объект window ).
- Модули всегда выполняются в строгом режимеstrict mode.
- Загрузка одного и того же модуля дважды в один и тот же файл не будет иметь никакого эффекта, поскольку модули выполняются только один раз.
- Работа с модулями требует серверной среды выполнения кода.
Модули по-прежнему широко используются при работе со сборщиками кода такими, как Webpack, для обеспечения поддержки браузерами дополнительных возможностей языка, а также доступны для использования непосредственно в браузерах.
Далее мы изучим еще несколько эффективных способов использования синтаксиса import и export .
Именованный экспорт
Как уже говорилось ранее, использование синтаксиса export позволяет выборочно импортировать именованные инструкции кода, которые экспортируются из модуля по имени. Например, рассмотрим следующую упрощенную версию файла functions.js :
Этот код позволяет импортировать функции sum и difference по их имени, используя фигурные скобки:
Также можно использовать синтаксис псевдонимов для переименования импортируемых функций. Эта возможность позволяет избежать конфликтов имен в импортирующем файле. В следующем примере функция sum будет переименована в add , difference — в subtract .
Вызов функции add() эквивалентен вызову функции sum() .
Используя * синтаксис, вы можете импортировать содержимое всего модуля в составе одного объекта. В этом случае функции sum и difference будут доступны как методы объекта mathFunctions :
Примитивные значения, выражения, определения функций , асинхронные функции , классы и экземпляры классов могут быть экспортированы, если у них есть свой идентификатор:
Теперь все экспортируемое можно успешно импортировать. Другой вид экспорта, с которым мы познакомимся в следующем разделе, известен как экспорт по умолчанию.
Экспорт по умолчанию
В предыдущих примерах мы успешно экспортировали несколько именованных инструкций кода, а затем импортировали их по отдельности или в составе одного объекта, в котором импортируемый код содержался в его свойствах и методах. Модули Javascript поддерживают возможность экспорта по умолчанию с использованием ключевого слова default . Экспорт по умолчанию не использует фигурные скобки и содержимое модуля будет импортироваться непосредственно в идентификатор с заданным вами именем.
Перепишем код файла functions.js в следующем виде:
В файле script.js импортируем функцию sum по умолчанию, как это показано в коде ниже:
Использование этой конструкции кода может привести к неожиданным ошибкам, поскольку нет ограничений на то, что вы можете назначить в качестве имени идентификатора экспорта по умолчанию, и что, соответственно, будете импортировать в инструкции импорта. В следующем примере из модуля по умолчанию экспортируется функция sum , а в переменную difference в инструкции импорта будет фактически импортирована функция sum :
По этой причине рекомендуется использовать именованный экспорт из модуля, так как экспорт по умолчанию не требует идентификатора для передачи в него экспортируемого кода. Экспортом же по умолчанию обычно экспортируются примитивные значения или анонимные функции. Ниже приведен пример объекта, экспортируемого по умолчанию:
Импортировать объект book можно следующим образом:
Точно так же в следующем примере демонстрируется экспорт по умолчанию анонимной стрелочной функции:
Функция импортируется из модуля файла script.js следующим образом:
Именованный экспорт и экспорт по умолчанию можно использовать совместно друг с другом, как в модуле из примера ниже, который экспортирует два именованных значения и одно значение (функцию) по умолчанию:
Импортируются эти переменные и функция следующим образом:
Теперь в скрипте доступны как значение по умолчанию, так и именованные значения.
Вывод
Методы использования модульного подхода программирования позволяют разделять код на отдельные компоненты, что делает ваш код более пригодным для повторного использования, согласованным, а также защищает глобальное пространство имен от загрязнения. Интерфейс модуля может быть реализован в нативном JavaScript с использованием ключевых слов import и export .
В этой статье вы узнали об истории модулей в JavaScript, как разделить файлы с кодом в JavaScript на несколько сценариев для имитации модульности, как обновить содержимое этих файлов с использованием модульного подхода и инструкций с ключевыми словами import и export , а также как использовать именованный экспорт и экспорт по умолчанию.
Читайте также: