如何引用加载当前运行的javascript的脚本元素?

情况是这样的。我有一个“主”脚本被加载在页面的高,第一件事下的头标签。

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" 
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
<head>
<script type="text/javascript" src="scripts.js"></script>

在“scripts.js”中有一个脚本,需要能够按需加载其他脚本。正常的方法不太适合我,因为我需要添加新的脚本而不引用HEAD标记,因为HEAD元素还没有完成渲染:

document.getElementsByTagName('head')[0].appendChild(v);

我要做的是引用加载当前脚本的脚本元素,这样我就可以在它之后将新的动态加载的脚本标记追加到DOM中。

<script type="text/javascript" src="scripts.js"></script>
loaded by scripts.js--><script type="text/javascript" src="new_script1.js"></script>
loaded by scripts.js --><script type="text/javascript" src="new_script2.js"></script>

当前回答

如何获取当前脚本元素:

1. 使用document.currentScript

文档。currentScript将返回当前正在处理的<script>元素。

<script>
var me = document.currentScript;
</script>

好处

简单明了。可靠的。 不需要修改脚本标记 使用异步脚本(defer & async) 使用动态插入的脚本

问题

不能在旧的浏览器和IE中工作。 <script type="module">

2. 按id选择脚本

给脚本一个id属性可以让您使用document.getElementById()从内部通过id轻松地选择它。

<script id="myscript">
var me = document.getElementById('myscript');
</script>

好处

简单明了。可靠的。 几乎得到了普遍支持 使用异步脚本(defer & async) 使用动态插入的脚本

问题

需要向脚本标记添加自定义属性 Id属性在某些边缘情况下可能会导致某些浏览器中的脚本出现奇怪的行为

3.使用data-*属性选择脚本

给脚本一个data-*属性可以让您轻松地从内部选择它。

<script data-name="myscript">
var me = document.querySelector('script[data-name="myscript"]');
</script>

与前一个选项相比,这几乎没有什么好处。

好处

简单明了。 使用异步脚本(defer & async) 使用动态插入的脚本

问题

需要向脚本标记添加自定义属性 HTML5和querySelector()在所有浏览器中都不兼容 与使用id属性相比,支持较少 将绕过<script>与id边缘情况。 如果页面上的另一个元素具有相同的数据属性和值,则可能会混淆。

4. 根据src选择脚本

不使用数据属性,你可以使用选择器根据源来选择脚本:

<script src="//example.com/embed.js"></script>

在embed.js:

var me = document.querySelector('script[src="//example.com/embed.js"]');

好处

可靠的 使用异步脚本(defer & async) 使用动态插入的脚本 不需要自定义属性或id

问题

不工作的本地脚本 会在不同的环境中造成问题,比如开发和生产 静止而脆弱。更改脚本文件的位置需要修改脚本 与使用id属性相比,支持较少 是否会导致问题,如果你加载相同的脚本两次

5. 遍历所有脚本以找到所需的脚本

我们还可以遍历每个脚本元素,并逐个检查以选择我们想要的元素:

<script>
var me = null;
var scripts = document.getElementsByTagName("script")
for (var i = 0; i < scripts.length; ++i) {
    if( isMe(scripts[i])){
      me = scripts[i];
    }
}
</script>

这让我们可以在不很好地支持querySelector()属性的旧浏览器中使用上述两种技术。例如:

function isMe(scriptElem){
    return scriptElem.getAttribute('src') === "//example.com/embed.js";
}

这继承了采用任何方法的优点和问题,但不依赖于querySelector(),因此可以在旧的浏览器中工作。

6. 获取最后执行的脚本

由于脚本是按顺序执行的,所以最后一个script元素通常是当前运行的脚本:

<script>
var scripts = document.getElementsByTagName( 'script' );
var me = scripts[ scripts.length - 1 ];
</script>

好处

简单。 几乎得到了普遍支持 不需要自定义属性或id

问题

不能使用异步脚本(defer & async) 不与动态插入的脚本一起工作

其他回答

这是一个利用文档的填充程序。如果存在CurrentScript,则返回到按ID查找脚本。

<script id="uniqueScriptId">
    (function () {
        var thisScript = document.CurrentScript || document.getElementByID('uniqueScriptId');

        // your code referencing thisScript here
    ());
</script>

如果您在每个脚本标记的顶部包含这个,我相信您将能够始终如一地知道哪个脚本标记正在被触发,并且您还能够在异步回调的上下文中引用脚本标记。

未经测试,所以如果你尝试了,给别人留下反馈。

只有当脚本没有“defer”或“async”属性时,才会按顺序执行。知道脚本标记的ID/SRC/TITLE属性之一也可以在这些情况下工作。Greg和Justin的建议都是正确的。

已经有人提议起草一份文件。WHATWG列表中的currentScript。

编辑:Firefox > 4已经实现了这个非常有用的属性,但它在IE11中不可用,我最后检查,只有在Chrome 29和Safari 8中可用。

编辑:没有人提到“文件”。但我相信下面可能是一个很好的跨浏览器的替代方案,以获得当前运行的脚本:

var me = document.scripts[document.scripts.length -1];

遵循以下简单步骤获取当前正在执行的脚本块的引用:

在脚本块中放入一些随机的唯一字符串(在每个脚本块中必须是唯一的/不同的) document.getElementsByTagName('script')的迭代结果,从每个内容中查找唯一的字符串(从innerText/textContent属性获得)。

示例(ABCDE345678是唯一的ID):

<script type="text/javascript">
var A=document.getElementsByTagName('script'),i=count(A),thi$;
for(;i;thi$=A[--i])
  if((thi$.innerText||thi$.textContent).indexOf('ABCDE345678'))break;
// Now thi$ is refer to current script block
</script>

顺便说一句,对于你的情况,你可以简单地使用老式的document.write()方法来包含另一个脚本。 正如你提到的DOM还没有被呈现,你可以利用浏览器总是以线性顺序执行脚本的事实(除了延迟的脚本将在以后呈现),所以你的文档的其余部分仍然是“不存在”的。 通过document.write()编写的任何内容都将放在调用者脚本之后。

原始HTML页面示例:

<!doctype html>
<html><head>
<script src="script.js"></script>
<script src="otherscript.js"></script>
<body>anything</body></html>

script.js的内容:

document.write('<script src="inserted.js"></script>');

渲染后,DOM结构将变成:

HEAD
  SCRIPT script.js
  SCRIPT inserted.js
  SCRIPT otherscript.js
BODY

我有这个,它在FF3, IE6和7中工作。按需加载脚本中的方法直到页面加载完成才可用,但这仍然非常有用。

//handle on-demand loading of javascripts
makescript = function(url){
    var v = document.createElement('script');
    v.src=url;
    v.type='text/javascript';

    //insertAfter. Get last <script> tag in DOM
    d=document.getElementsByTagName('script')[(document.getElementsByTagName('script').length-1)];
    d.parentNode.insertBefore( v, d.nextSibling );
}

可能最简单的方法就是给你的scrip标签一个id属性。