溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

如何在JavaScript中使用私有類字段

發(fā)布時(shí)間:2021-04-30 14:44:49 來源:億速云 閱讀:192 作者:Leah 欄目:開發(fā)技術(shù)

如何在JavaScript中使用私有類字段?很多新手對此不是很清楚,為了幫助大家解決這個難題,下面小編將為大家詳細(xì)講解,有這方面需求的人可以來學(xué)習(xí)下,希望你能有所收獲。

javascript是一種什么語言

javascript是一種動態(tài)類型、弱類型的語言,基于對象和事件驅(qū)動并具有相對安全性并廣泛用于客戶端網(wǎng)頁開發(fā)的腳本語言,同時(shí)也是一種廣泛用于客戶端Web開發(fā)的腳本語言。它主要用來給HTML網(wǎng)頁添加動態(tài)功能,現(xiàn)在JavaScript也可被用于網(wǎng)絡(luò)服務(wù)器,如Node.js。

JavaScript私有類字段和隱私需求

在過去,JavaScript 沒有保護(hù)變量不受訪問的原生機(jī)制,當(dāng)然除非是典型閉包。

閉包是 JavaScript 中許多類似于私有模式(如流行的模塊模式)的基礎(chǔ)。但是,近年來 ECMAScript 2015 類被使用后,開發(fā)人員感到需要對類成員的隱私進(jìn)行更多控制。

類字段提案(在撰寫本文時(shí)處于第 3 階段)試圖通過引入私有類字段來解決問題。

讓我們看看它們是什么樣子的。

一個 JavaScript 私有類字段的例子

這是一個帶有私有字段的 JavaScript 類,請注意,與“公有”成員不同,每個私有字段必須在訪問前進(jìn)行聲明:

class Person {
  #age;
  #name;
  #surname;
  constructor(name, surname, age) {
    this.#name = name;
    this.#surname = surname;
    this.#age = age;
  }
  getFullName() {
    return `${this.#name} + ${this.#surname}`;
  }
}

無法從類的外部訪問私有類字段:

class Person {
  #age;
  #name;
  #surname;
  constructor(name, surname, age) {
    this.#name = name;
    this.#surname = surname;
    this.#age = age;
  }
  getFullName() {
    return `${this.#name} + ${this.#surname}`;
  }
}
const marta = new Person("Marta", "Cantrell", 33);
console.log(marta.#age); // SyntaxError

這是真正的“隱私”。如果你會一點(diǎn) TypeScript,可能會問“原生”私有字段與TypeScript 中的 private 修飾符有什么共同點(diǎn)。

好吧,答案是:沒有。但是為什么?

TypeScript 中的 private 修飾符

有著傳統(tǒng)編程語言背景的開發(fā)人員應(yīng)該熟悉 TypeScript 中的 private 修飾符。簡而言之,此關(guān)鍵字的目的是拒絕從類的外部訪問類成員。

但是請不要忘記,TypeScript 是處于 JavaScript 之上的一層,并且 TypeScript 編譯器應(yīng)該剝離所有花里胡哨的 TypeScript 注釋,包括private。

這意味著下面的類做不到你想要的工作:

class Person {
  private age: number;
  private name: string;
  private surname: string;
  constructor(name: string, surname: string, age: number) {
    this.name = name;
    this.surname = surname;
    this.age = age;
  }
  getFullName() {
    return `${this.name} + ${this.surname}`;
  }
}
const liz = new Person("Liz", "Cantrill", 31);
// @ts-ignore
console.log(liz.age);

如果沒有//@ts-ignore,在訪問liz.age時(shí)僅會在 TypeScript中引發(fā)錯誤,但是在編譯之后,你將會得到下面的 JavaScript代碼:

"use strict";
var Person = /** @class */ (function () {
    function Person(name, surname, age) {
        this.name = name;
        this.surname = surname;
        this.age = age;
    }
    Person.prototype.getFullName = function () {
        return this.name + " + " + this.surname;
    };
    return Person;
}());
var liz = new Person("Liz", "Cantrill", 31);
console.log(liz.age); // 31

與預(yù)期的一樣,我們可以從控制臺輸出liz.age。這里的主要觀點(diǎn)是 TypeScript 中的 private 不是那么私有,并且僅在 TypeScript 級別才感到方便,而不是“真正的隱私”。

接下來我們開始討論:TypeScript 中的“原生”私有類字段。

TypeScript 中的私有類字段

TypeScript 3.8 將支持 ECMAScript 私有字段,千萬別和TypeScript private 修飾符混淆。

這是在 TypeScript 中具有私有類字段的類:

class Person {
    #age: number;
    #name: string;
    #surname: string;
    constructor(name:string, surname:string, age:number) {
        this.#name = name;
        this.#surname = surname;
        this.#age = age;
    }
    getFullName() {
        return `${this.#name} + ${this.#surname}`;
    }
}

除了類型注釋外,與原生 JavaScript 沒什么不同。無法從外部訪問成員。但是 TypeScript 中私有字段的真正問題在于它們在后臺使用了 WeakMap。

要編譯此代碼,我們需要調(diào)整 tsconfig.json 中的目標(biāo)編譯版本,該版本最低必須是ECMAScript 2015:

{
  "compilerOptions": {
    "target": "es2015",
    "strict": true,
    "lib": ["dom","es2015"]
  }
}

這可能會出現(xiàn)問題,具體取決于目標(biāo)瀏覽器,除非你打算為 WeakMap 提供polyfill,否則,如果只是為了編寫精美的新語法,工作量就變得太多了。

JavaScript 中總是存在這種緊張關(guān)系,你確實(shí)想使用新語法,但另一方面,你不想由于大量的 polyfill 影響使用戶體驗(yàn)。

另一方面,即使你希望將其發(fā)布到較新的瀏覽器中,也不必?fù)?dān)心私有類字段。最起碼現(xiàn)在是這樣。甚至Firefox都沒有實(shí)施該建議。

看完上述內(nèi)容是否對您有幫助呢?如果還想對相關(guān)知識有進(jìn)一步的了解或閱讀更多相關(guān)文章,請關(guān)注億速云行業(yè)資訊頻道,感謝您對億速云的支持。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI