导入值与其导出值是否断开连接,是否仍然是只读的

Are import values disconnected from their export values still read-only?

本文关键字:是否 仍然是 只读 连接 断开 导入      更新时间:2023-09-26

给出的模块结构如下:

// module A:
export let a = 1; // named export
export function inc() { a++; } // named export
// module B:
let b = 1;
export default b; // default export (equivalent to `export default 1`)
export function inc() { b++; } // named export
// module C:
let c = {};
export default c; // default export
// module E:
import a, {inc as incA} from "./A";
import b, {inc as incB} from "./B";
import c from "./C";
incA();
console.log(a); // logs 2, because "a" has a live connection to the export value
a++; // Error (because a is a live read-only view on the export)
incB();
console.log(b); // logs 1, because "b" is disconnected from the export value
b++; // Does this throw an error as well?
c.prop = true; // I think mutations are always allowed, right?
c = {}; // but are reassignment allowed too?

如果我有一个表达式的默认导出(export default bexport default 1),相应的导入将与该导出值断开连接。考虑到这一点,这样的导入仍然是只读的吗?就是说,我可以重新分配ac吗?

导入绑定始终是只读的,参见规范中的抽象CreateImportBinding操作,步骤5:

  • envRec中为N创建一个不可变的间接绑定,该绑定引用MN2作为其目标绑定,并记录该绑定已初始化。
  • (我强调)

    ModuleDeclarationInstantiation在处理模块的导入项时使用该操作。

    :

    b++; // Does this throw an error as well?
    

    是,b为只读。

    c.prop = true; // I think mutations are always allowed, right?
    

    如果导出的对象允许,可以。

    c = {}; // but are reassignment allowed too?
    

    No, c为只读。


    在评论中,你说:

    但是当不再存在活绑定时,将这些变量设置为只读是没有意义的

    记住它们不是变量是有用的,它们是绑定。虽然变量是一种绑定,但并不是所有的绑定都是变量(即使在ES5和更早的版本中也是如此)。

    关于它们在无关紧要的情况下是只读的,请记住涉及到两层绑定:

    1. 源模块中导出的动态绑定。
    2. import在消费模块中的动态绑定。

    当#1的值不变时,为了使#2可写,导入机制必须知道这一点,但该信息不在模块的导出中。导出只给出导出绑定的名称。

    此外,拥有可变导入绑定和不可变导入绑定更难以理解,实现起来也更复杂。(从样式的角度来看,重新分配导入的绑定也是令人困惑的。)

    同样重要的是要记住,它们是活动绑定的事实意味着import 的值可能会改变。假设:

    mod1.js:

    export let foo = 41;
    export function incrementFoo() {
        ++foo;
    };
    

    mod2.js:

    import { foo, incrementFoo } from "./mod1.js";
    console.log(foo); // 41
    incrementFoo();
    console.log(foo); // 42 <== it changed
    

    在这种特殊情况下,是mod2.js中的代码引起了更改(通过调用incrementFoo),但它不必是。这可能是因为mod1.js中发生了一些与时间相关的事件,使值发生了变化,或者是其他模块调用mod1.js的结果,等等。

    但是因为mod2.jsfoomod1.jsfoo的活绑定,mod2.js看到了这个变化