Cory Rylan

My name is , Google Developer Expert, Speaker, Software Developer. Building Design Systems and Web Components.

Follow @coryrylan
TypeScript

Private Methods and Properties in TypeScript Classes

Cory Rylan

- 3 minutes

JavaScript has drastically improved as a language over the past few years. TypeScript provides some nice features on top of the JavaScript such as static typing. In this post, we are going to cover one of these TypeScript features, the private keyword.

ES2015 Classes

Before we dive into TypeScript's private feature let's do a quick recap of JavaScript classes. In the new ES2015 standard of JavaScript we get a Object Oriented Class style syntax that looks like the following,

export class Person {
constructor(name) {
this.name = name;
}

sayHello() {
console.log(`Hello, my name is ${this.name}!`);
}
}

const cory = new Person('Cory');
cory.sayHello(); // Hello, my name is Cory!

In this example we are using pure JavaScript no TypeScript syntax or features are being used. JavaScript classes can be exported and used in other JavaScript modules. JavaScript classes also have constructors, properties, and methods similar to most Class-based languages we see today. Unfortunately, in the current version of JavaScript, there is no support for private properties or private methods yet. In JavaScript all class instance properties and methods are public.

TypeScript Private Properties

Using TypeScript, we can add private functionality into our classes. What are private properties or methods? A private property of method can only be accessed or called from the class instance itself. Let's take a look at an example private property.

export class Person {
// declare our property types
firstName: string;
lastName: string;
private _age: number;

// when accessing the age property return the private _age
// getters and setters are part of the JavaScript Class syntax
get age() {
return this._age;
}

constructor(firstName: string, lastName: string, age: number) {
this.firstName = firstName;
this.lastName = lastName;
this._age = age;
}

sayHello() {
console.log(`Hello, my name is ${this.firstName} ${this.lastName}!`);
}

// Only this method can update the private _age
addOneYear() {
this._age = this._age + 1;
}
}

const cory = new Person('Cory', 'Rylan', 100);
cory.addOneYear();
console.log(cory.age); // 101

cory._age = 200; // error: Property '_age' is private and only accessible within class 'Person'.
console.log(cory._age); // error: Property '_age' is private and only accessible within class 'Person'.
cory.age = 200; // error: Cannot assign to 'age' because it is a constant or a read-only property.

In this example, we are using a typical pattern of a private property to control how a property is updated. In our use case, it is valid to increase the age of a person, but you cannot set it to a random value or a younger age. To enforce this, we create a private property _age. The _age property is a property that will be only available internally to the class. For example, if I try to set _age I get an error because it is private to the class. If I try to read _age I also get the same error

We can see the private hides the property from the user/consumer of the class. If I try to set cory.age we also get an error because I defined only a get for the age property with no set so this property can only be read-only. We can go one step further and refactor our constructor a little bit more.

TypeScript Constructor Assignment

In this example, we can simplify our constructor parameters by combining the declaration and assignment into a single statement.

export class Person {
private _age: number;

get age() {
return this._age;
}

constructor(public firstName: string, public lastName: string, age: number) {
this._age = age;
}

sayHello() {
console.log(`Hello, my name is ${this.firstName} ${this.lastName}!`);
}

addOneYear() {
this._age = this._age + 1;
}
}

const cory = new Person('Cory', 'Rylan', 100);
cory.sayHello();

We can prefix the constructor parameters with the public or private keyword to automatically have TypeScript assign the parameter as a property of the class. In this example, this removes the unnecessary declaration and assignment of both firstName and lastName.

TypeScript Private Methods

Methods can also be private which is useful for hiding implementation detail of how a Class works to the user of the Class. Let's take a look at a minimal example.

export class Person {
private _age: number;

get age() {
return this._age;
}

constructor(public firstName: string, public lastName: string, age: number) {
this._age = age;
}

sayHello() {
this.log(`Hello, my name is ${this.firstName} ${this.lastName}!`);
}

addOneYear() {
this._age = this._age + 1;
}

private log(message) {
console.log(message);
}
}

const cory = new Person('Cory', 'Rylan', 100);
cory.sayHello(); // Hello, my name is Cory Rylan!
cory.log('hi'); // error: Property 'log' is private and only accessible within class 'Person'.

In this example, we created a private method log(). Log can only be called by other methods in our class. You can see above if I try to call log directly, we get a TypeScript error. Private properties and methods can help enforce logic for when data is updated and also enforce encapsulation of our classes. Check out the full working demo in the link below!

View Demo Code   
Twitter Facebook LinkedIn Email
 

No spam. Short occasional updates on Web Development articles, videos, and new courses in your inbox.

Related Posts

Preact JS

How to use Web Components in Preact and TypeScript

Learn how to use Web Components with TSX TypeScript and Preact components.

Read Article
React JS

How to use Web Components with TypeScript and React

Learn how to use Web Components with TSX TypeScript and React components.

Read Article
Lit Web Components

Using Event Decorators with lit-element and Web Components

Learn how to make an event decorator to make it easy to emit custom events with improved type safety in Lit Web Components.

Read Article