在Javascript中,如何在不更改函数的情况下检查函数是否处于严格模式*

In Javascript, how can I check if a function is in strict mode *without changing the function*?

本文关键字:函数 检查 情况下 是否 模式 于严格 Javascript      更新时间:2023-09-26

我想写一个测试套件,以确保某些给定的函数使用严格模式。它们有很多,手动检查它们似乎是一件烦人的事。

类似问题的答案使用函数定义上的正则表达式进行检查。然而,我相信这会错误地检测到被测试的函数位于具有"use strict"或文件级"use strict."声明的函数内部的情况。答案是"严格使用"是预先准备好的,但在我的环境(Mozilla Rhino)中,情况并非如此:

$ cat strict_sub.js
"use strict";
var strict_function = function() {
    not_a_real_global = "foo";
};
print(strict_function);
$ rhino strict_sub.js 
function () {
    not_a_real_global = "foo";
}

我觉得答案是"不",但难道没有办法内省一个函数,看看它是否被解析并被发现是严格模式吗?

更新:@Amy建议的一种方法是解析函数的源代码来找出它。如果函数有一个使用严格声明(尽管这很乏味),这是有效的,但如果它的严格模式是从程序级别传播,则无效;在这种情况下,我们必须将AST提升到Program级别,并检查是否存在use strict。为了使其健壮,我们必须实现use strict-传播的所有规则,解释器已经在某个地方有了这些规则。

(在SpiderMonkey:中测试

function f() {
  "use strict";
}
var fast1 = Reflect.parse(f.toString());
var first_line = fast1.body[0].body.body[0].expression;
print(first_line.type === 'Literal' && first_line.value === 'use strict'); //true

)

严格模式函数确实有一个"中毒"的.caller.arguments属性(还有ES5,额外),所以您可以测试它:

function isStrict(fn) {
    if (typeof fn != "function")
        throw new TypeError("expected function");
    try {
        fn.caller; // expected to throw
        return false;
    } catch(e) {
        return true;
    }
}