猿问

我可以将 javascript 模块与“常规”javascript 混合使用吗?

我有一个页面设置,它引用了几个 javascript 文件,并在页面中内嵌了一些 javascript,这是在加载页面时生成的。

其中一个 JavaScript 文件包含来自第三方库的类。

另一个 javascript 文件包含与页面相关的各种变量、事件处理程序和函数,它们使用第三方库 ( var v = new thirdPartyClass(parameters);) 中的类。

第三方最近更新了他们的库并转而使用模块,因此现在他们不再提供可用的类,而是提供了可用的类导出。这是我第一次接触js模块。

因此,据我了解,现在我必须从模块导入该类,而不是包含 js 文件。当我尝试这样做时,我收到一个控制台错误,只有模块可以导入。我添加type="module"<script>我的js文件的标签(它已经添加到第三方js脚本标签),然后我可以导入他们的类,但现在我的函数无法从页面(function Uncaught ReferenceError: myFunction is not defined)访问。我不知道活动是否有效,因为它不会再发展到那么远了。

有什么方法可以使用他们的新导出类,而不必将我的 js 文件完全重组为模块并更改页面以使用它?


莫回无
浏览 127回答 4
4回答

杨__羊羊

您可以创建一个脚本类型模块,导入所需的对象并将它们放入窗口中。{您的对象的名称}如下:<script type="module">&nbsp; &nbsp; import { theItem } from "/path/to/module.js";&nbsp; &nbsp; window.theItem = theItem;</script>其余代码可以在常规 js 中。

慕容3067478

您无法使用 . 在模块中创建全局变量var。要访问该函数,您可以将其从文件中导出:export myFunction;并将其导入到您需要的文件中:import {myFunction} from "./relativeFilePath";或者,如果您想从模块内创建全局变量,您可以将该变量直接放入对象中window:window.myFunction = function() {&nbsp; //code};

呼唤远方

因此,据我了解,现在我必须从模块导入该类,而不是包含 js 文件。当我尝试这样做时,我收到一个控制台错误,只有模块可以导入。对于传统的静态语句来说确实如此import。但是,值得注意的是,动态import()运算符(另一种导入模块的异步方式)可用于将模块导入到经典 javascript 文件以及其他模块中。MDN注释:该import()语法通常称为动态导入,是一种类似函数的表达式,允许将 ECMAScript 模块异步动态加载到潜在的非模块环境中[...]和:[...]您可能需要使用动态导入:当您处于非模块环境中时(例如 [...]脚本文件)。例子:// /my-classic-javascript-file.jsconst myAsyncFunction = async () => {  const { myImportedClass } = await import('/path/to/module.js');}

冉冉说

属性window设置器解决方案:<script type="module">&nbsp; &nbsp;import Dms from 'https://cdn.jsdelivr.net/npm/geodesy@2.4.0/dms.js';&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;&nbsp; &nbsp;alert(Dms.toLat(-3.62, 'dms'));&nbsp; &nbsp;window.Dms = Dms;</script><script>&nbsp; let Dms;&nbsp; Object.defineProperty(window, 'Dms', {&nbsp; &nbsp; get(){&nbsp; &nbsp; &nbsp; return Dms;&nbsp; &nbsp; },&nbsp; &nbsp; set(val){&nbsp; &nbsp; &nbsp; Dms = val;&nbsp; &nbsp; &nbsp; window.dispatchEvent(new Event('dms-set'));&nbsp; &nbsp; }&nbsp; });&nbsp;&nbsp;&nbsp; window.addEventListener('dms-set', () => alert('SECOND TIME: ' + Dms.toLat(-3.62, 'dms')));&nbsp;&nbsp;</script>一些承诺的解决方案:<script type="module">&nbsp; &nbsp;import Dms from 'https://cdn.jsdelivr.net/npm/geodesy@2.4.0/dms.js';&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;&nbsp; &nbsp;alert(Dms.toLat(-3.62, 'dms'));&nbsp; &nbsp;window.Dms = Dms;</script><script>&nbsp; promisifyGlobal('Dms');&nbsp;&nbsp;&nbsp; Dms.then(Dms => alert('SECOND TIME: ' + Dms.toLat(-3.62, 'dms')));&nbsp; function promisifyGlobal(name){&nbsp; &nbsp; let global, promise = new Promise(resolve => {&nbsp; &nbsp; &nbsp; Object.defineProperty(window, name, {&nbsp; &nbsp; &nbsp; &nbsp; get(){&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; return promise;&nbsp; &nbsp; &nbsp; &nbsp; },&nbsp; &nbsp; &nbsp; &nbsp; set(val){&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; global = val;&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; resolve(global);&nbsp; &nbsp; &nbsp; &nbsp; }&nbsp; &nbsp; &nbsp; });&nbsp; &nbsp; });&nbsp; }</script>您可以轻松地动态导入到非模块脚本中:<script>&nbsp; (async () => {&nbsp; &nbsp;&nbsp;&nbsp; &nbsp; const {default: Dms} = await import('https://cdn.jsdelivr.net/npm/geodesy@2.4.0/dms.js');&nbsp; &nbsp; alert(Dms.toLat(-3.62, 'dms'));&nbsp;&nbsp;&nbsp; })();</script>最后是一些疯狂的通用解决方案:<script type="module">&nbsp; import Dms from 'https://cdn.jsdelivr.net/npm/geodesy@2.4.0/dms.js';&nbsp;&nbsp; alert(Dms.toLat(-3.62, 'dms'));</script><script>&nbsp; // this script will handle all imports&nbsp;&nbsp; // tweak to import several exports and named exports&nbsp; for(const script of document.querySelectorAll('script[type=module]')) {&nbsp; &nbsp;&nbsp;&nbsp; &nbsp; const regex = /import\s+([^\s]+)\s+from\s+["\']([^"\']+)/g;&nbsp; &nbsp; let matches;&nbsp; &nbsp; while(matches = regex.exec(script.textContent)){&nbsp; &nbsp; &nbsp; const [, name, url] = matches;&nbsp; &nbsp; &nbsp; const src = `import ${name} from '${url}'; export default ${name};`;&nbsp; &nbsp; &nbsp; window[name] = new Promise(resolve => {&nbsp; &nbsp; &nbsp; &nbsp; import(`data:text/javascript, ${src}`).then(({default:imported}) => {&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; setTimeout(() => resolve(imported));&nbsp; &nbsp; &nbsp; &nbsp; });&nbsp; &nbsp; &nbsp; });&nbsp; &nbsp; }&nbsp; }</script><script>&nbsp; Dms.then(Dms => alert('SECOND TIME: ' + Dms.toLat(-3.62, 'dms')));</script>
随时随地看视频慕课网APP

相关分类

JavaScript
我要回答