詳解JavaScript私有類字段和TypeScript私有修飾符
JavaScript私有類字段和隱私需求
在過去,JavaScript 沒有保護變量不受訪問的原生機制,當然除非是典型閉包。
閉包是 JavaScript 中許多類似于私有模式(如流行的模塊模式)的基礎(chǔ)。但是,近年來 ECMAScript 2015 類被使用后,開發(fā)人員感到需要對類成員的隱私進行更多控制。
類字段提案(在撰寫本文時處于第 3 階段)試圖通過引入私有類字段來解決問題。
讓我們看看它們是什么樣子的。
一個 JavaScript 私有類字段的例子
這是一個帶有私有字段的 JavaScript 類,請注意,與“公有”成員不同,每個私有字段必須在訪問前進行聲明:
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
這是真正的“隱私”。如果你會一點 TypeScript,可能會問“原生”私有字段與TypeScript 中的 private 修飾符有什么共同點。
好吧,答案是:沒有。但是為什么?
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時僅會在 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。這里的主要觀點是 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 中的目標編譯版本,該版本最低必須是ECMAScript 2015:
{ "compilerOptions": { "target": "es2015", "strict": true, "lib": ["dom","es2015"] } }
這可能會出現(xiàn)問題,具體取決于目標瀏覽器,除非你打算為 WeakMap 提供polyfill,否則,如果只是為了編寫精美的新語法,工作量就變得太多了。
JavaScript 中總是存在這種緊張關(guān)系,你確實想使用新語法,但另一方面,你不想由于大量的 polyfill 影響使用戶體驗。
另一方面,即使你希望將其發(fā)布到較新的瀏覽器中,也不必擔心私有類字段。最起碼現(xiàn)在是這樣。甚至Firefox都沒有實施該建議。
以上就是詳解JavaScript私有類字段和TypeScript私有修飾符的詳細內(nèi)容,更多關(guān)于JavaScript私有類字段和TypeScript私有修飾符的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
js中關(guān)于new Object時傳參的一些細節(jié)分析
這里討論給Object傳參時,其內(nèi)部的處理。參考:ECMA262 V5 15.2.2.12011-03-03微信小程序購物車、父子組件傳值及calc的注意事項總結(jié)
這篇文章主要給大家介紹了關(guān)于微信小程序購物車、父子組件傳值及calc的注意事項的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2018-11-11java與javascript之間json格式數(shù)據(jù)互轉(zhuǎn)介紹
對象轉(zhuǎn)為字符串:通過JSON.encode方法,這個是json.js里面的方法,引入到當前文件就可以了,下面整理的比較詳細一點,感興趣的朋友不要錯過2013-10-10Javascript在IE下設(shè)置innerHTML時出現(xiàn)未知的運行時錯誤的解決方法
程序代碼思路大致如下,想動態(tài)的改變tr標簽的innerHTML內(nèi)容,但是IE卻報未知的運行時錯誤2011-01-01IE6中使用position導致頁面變形的解決方案(js代碼)
IE6中使用position導致頁面變形,影響用戶體驗,通過搜索可以通過js來實現(xiàn)。2011-01-01分享5個小技巧讓你寫出更好的 JavaScript 條件語句
在使用 JavaScript 時,我們常常要寫不少的條件語句。這里有五個小技巧,可以讓你寫出更干凈、漂亮的條件語句。需要的朋友跟隨小編一起學習吧2018-10-10javascript設(shè)計模式 – 原型模式原理與應(yīng)用實例分析
這篇文章主要介紹了javascript設(shè)計模式 – 原型模式,結(jié)合實例形式分析了javascript原型模式相關(guān)概念、原理、應(yīng)用場景及操作注意事項,需要的朋友可以參考下2020-04-04JavaScript 用Node.js寫Shell腳本[譯]
你懂JavaScript嗎?你需要寫一個Shell腳本嗎?那么你應(yīng)該試一下Node.js,它很容易安裝,而且很適合通過寫Shell腳本來學習它2012-09-09