JavaScript 如何创建私有变量

JavaScript 如何创建私有变量

闭包、符号、弱映射、类私有字段和代理是几种可以用来在JavaScript中创建私有变量的技术。每种技术都有优点和缺点,因此选择最适合您要求的技术非常关键。

和许多其他编程语言一样,JavaScript也有私有和公共变量的概念。私有变量只能被同一作用域中的代码访问和更改,而公共变量可以被任何代码访问和更改。让我们来看一下在JavaScript中创建私有变量的不同技术。

使用闭包

闭包方法是在JavaScript中创建私有变量的一种方法。如果一个函数可以访问其父函数作用域中定义的变量,即使父函数已经执行并返回,那么这个函数就是一个闭包。我们可以在函数内部定义一个变量,使其成为一个只能被该函数内部的代码访问的私有变量。

示例

<html>
<body>
   <div id="demo"></div>
   <script>
      function createPrivateVariable() {
         let privateVariable = "This is a private variable";
         return {
            getValue: function () {
               return privateVariable;
            },
            setValue: function (value) {
               privateVariable = value;
            },
         };
      }
      let privateVar = createPrivateVariable();
      document.getElementById("demo").innerHTML = privateVar.getValue();
   </script>
</body>
</html>

在上面的示例中,函数 createPrivateVariable 返回一个具有方法 getValuesetValue 的对象。这些方法可以检索或更改父函数中声明的 privateVariable 的值,因为它们可以访问它。如果尝试从函数外部访问privateVariable会导致引用错误。

使用Symbol数据类型

使用Symbol数据类型是创建私有变量的第二种方法。符号可以用作属性键,因为它们是不同的、非字符串的标识符。由于它们是唯一的,所以不能被外部程序轻易访问或修改。

let privateVariable = Symbol();
let obj = {
   [privateVariable]: "This is a private variable"
};
console.log(obj[privateVariable]);

示例

我们可以按照上面的代码使用如下:

<html>
<body>
   <div id="demo"></div>
   <script>
      let privateVar = Symbol();
      let obj = {
         [privateVar]: "This is a private variable"
      };
      Object.defineProperty(obj, 'getValue', {
         get: function() {
            return obj[privateVar];
         }
      });
      document.getElementById("demo").innerHTML = obj.getValue;
   </script>
</body>
</html>

在这个示例中,定义了一个名为privateVariable的Symbol,并将其用作对象的属性键。因为它是一个Symbol,所以不能使用点符号获取属性的值,但可以通过使用方括号符号在对象中访问它。

使用WeakMaps

WeakMaps可以用作构造私有变量的第三种方法。JavaScript引擎只对WeakMap中的键值对进行弱引用,这样可以将对象与键相关联。这使得误保持对私有变量的引用更加困难,因为如果没有其他对键的引用,垃圾回收器将销毁键值对。

示例

<html>
<body>
   <div id="demo"></div>
   <script>
      let privateVariables = new WeakMap();
      let obj = {};
      privateVariables.set(obj, "This is a private variable");
      document.getElementById("demo").innerHTML = privateVariables.get(obj);
   </script>
</body>
</html>

在这个示例中,我们创建了一个名为privateVariables的WeakMap,用于保存对象的私有变量。使用get()方法可以在使用set()方法将私有变量与对象连接后获取私有变量。但是,私有变量不能从对象形成的作用域外部访问,因为只有在有对象的引用时才能访问。

使用面向对象的类语法

也可以使用面向对象的类语法来创建私有变量。JavaScript的class关键字使我们能够定义类,类可以作为对象的模板。一个类可以用#符号在变量名前定义一个私有变量;这是一个实验性功能,表示一个私有变量。目前不建议在生产代码中使用此功能,因为它目前并没有广泛支持。

示例

<html>
<body>
   <p id="output"></p>
   <script>
      class MyClass {
         #privateVariable = "This is a private variable";
         getValue() {
            return this.#privateVariable;
         }
         setValue(value) {
            this.#privateVariable = value;
         }
      }
      let obj = new MyClass();
      document.getElementById("output").innerHTML = obj.getValue(); // "This is a private variable"
      obj.setValue("New value");
      document.getElementById("output").innerHTML = obj.getValue(); // "New value"
   </script>
</body>
</html>

在这个示例中,一个名为 MyClass 的类被构建,使用了私有变量 #privateVariablegetValuesetValue 。如果类外的方法尝试访问只能通过类内方法访问的私有变量,将会引发引用错误。

使用代理对象

最后,使用代理对象是构建私有变量的另一种选择。一个可以用来拦截或改变其他对象行为的对象被称为代理。你可以通过将对象封装在代理中,使私有变量只能被具有代理的代码访问。

示例

<html>
<body>
   <div id="demo"></div>
   <script>
      let privateVariable = "This is a private variable-proxy obj.";
      let handler = {
         get: function (target, name) {
            if (name === "privateVariable") {
               return privateVariable;
            }
         },
         set: function (target, name, value) {
            if (name === "privateVariable") {
               privateVariable = value;
            }
         },
      };
      let obj = new Proxy({}, handler);
      document.getElementById("demo").innerHTML = obj.privateVariable;
   </script>
</body>
</html>

在这个示例中,我们建立了一个带有getter和setter方法的Proxy处理程序。这些方法可以获取或更改在Proxy之外定义的私有变量的值,并对其进行访问。但是,由于引用错误,私有变量无法从Proxy外部访问。

Camera课程

Python教程

Java教程

Web教程

数据库教程

图形图像教程

办公软件教程

Linux教程

计算机教程

大数据教程

开发工具教程