Detailed explanation of JavaScript private class fields and TypeScript private modifiers

Detailed explanation of JavaScript private class fields and TypeScript private modifiers

JavaScript private class fields and the need for privacy

In the past, JavaScript had no native mechanism for protecting variables from being accessed, except of course with classic closures.

Closures are the basis for many privacy-like patterns in JavaScript, such as the popular module pattern. However, with the use of ECMAScript 2015 classes in recent years, developers have felt the need for more control over the privacy of class members.

The class fields proposal (at stage 3 at the time of writing) attempts to solve the problem by introducing private class fields.

Let's see what they look like.

A JavaScript private class field example

Here is a JavaScript class with private fields. Note that, unlike "public" members, each private field must be declared before being accessed:

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

Private class fields cannot be accessed from outside the class:

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

This is true "privacy". If you know a bit of TypeScript, you might be wondering what "native" private fields have in common with the private modifier in TypeScript.

Well, the answer is: no. But why?

The private modifier in TypeScript

Developers coming from a background in traditional programming languages ​​should be familiar with the private modifier in TypeScript. In short, the purpose of this keyword is to deny access to class members from outside the class.

But don't forget that TypeScript is a layer on top of JavaScript, and the TypeScript compiler should strip away all the fancy TypeScript comments, including private.

This means that the following class will not do what you want:

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);

Without //@ts-ignore, accessing liz.age would only cause an error in TypeScript, but after compilation, you would get the following JavaScript code:

"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

As expected, we can output liz.age to the console. The main point here is that private in TypeScript is not so private, and only feels convenient at the TypeScript level, not "real privacy".

Now let’s move on to: “native” private class fields in TypeScript.

Private class fields in TypeScript

TypeScript 3.8 will support ECMAScript private fields, not to be confused with the TypeScript private modifier.

Here is a class with private class fields in 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}`;
    }
}

Apart from the type annotations, it is no different from native JavaScript. Members cannot be accessed externally. But the real problem with private fields in TypeScript is that they use WeakMap under the hood.

To compile this code, we need to adjust the target compilation version in tsconfig.json, which must be at least ECMAScript 2015:

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

This may be problematic depending on the target browsers, and unless you plan to provide a polyfill for WeakMap, it becomes too much work just to write fancy new syntax.

There's always this tension in JavaScript where you do want to use the new syntax, but on the other hand, you don't want to degrade the user experience with lots of polyfills.

On the other hand, even if you want to release it to newer browsers, you don't have to worry about private class fields. At least that's the case now. Not even Firefox has implemented the suggestion.

The above is a detailed explanation of JavaScript private class fields and TypeScript private modifiers. For more information about JavaScript private class fields and TypeScript private modifiers, please pay attention to other related articles on 123WORDPRESS.COM!

You may also be interested in:
  • The connection between JavaScript and TypeScript
  • JS Decorator Pattern and TypeScript Decorators
  • What is the difference between python and js in getting MD5 hash
  • Detailed explanation of how to use js to implement md5 encryption in Easy Language
  • Introducing MD5 checksums in TypeScript and JavaScript projects

<<:  Installation tutorial of mysql5.7.21 decompression version under win10

>>:  How to configure static network connection in Linux

Recommend

How to display texture at the position of swipe in CocosCreator

Table of contents 1. Project requirements 2. Docu...

Solve the group by query problem after upgrading Mysql to 5.7

Find the problem After upgrading MySQL to MySQL 5...

Example usage of JavaScript tamper-proof object

Table of contents javascript tamper-proof object ...

GDB debugging MySQL actual combat source code compilation and installation

Download source code git clone https://github.com...

Detailed Analysis of the Selection of MySQL Common Index and Unique Index

Suppose a user management system where each perso...

How to implement logic reuse with Vue3 composition API

Composition API implements logic reuse steps: Ext...

Detailed explanation of the use of Arguments object in JavaScript

Table of contents Preface Basic Concepts of Argum...

How to set background color and transparency in Vue

Background color and transparency settings As sho...

HTML table mouse drag sorting function

Effect picture: 1. Import files <script src=&q...

Nginx request limit configuration method

Nginx is a powerful, high-performance web and rev...

js memory leak scenarios, how to monitor and analyze them in detail

Table of contents Preface What situations can cau...

An example of vertical centering of sub-elements in div using Flex layout

1. Flex is the abbreviation of Flexible Box, whic...

mysql5.6.zip format compressed version installation graphic tutorial

Preface: MySQL is a relational database managemen...