Classes in TypeScript have a few extra features compared to JS classes, mostly related to type-checking.
- You can declare uninitialized properties on the class body; Those don't generate JS code, they just declare types for checking.
- If a declared property is not initialized on the constructor, or directly, TS will complain. To fix this you can either declare a property as optional (append
?
) or assume it is not null (append!
).
class Foo {
constructor(name: string) {
this.name = name
}
name: string
hasBar?: string
certainlyNotNull!: number
}
The operator
!
can be postponed to a concrete identifier to tell TypeScript that you assume it will not be null.
Ex: Ifperson
is of typePerson | undefined
, thenperson!
will be of typePerson
.
- Access modifiers (
private
,protected
andpublic
) are a thing; Yet again, they only serve as hints to the type-checker. Aprivate
declared property will still be emitted and visible in JS code. - Class fields can be initialized in-body (same as JS, recent-y proposal);
class Foo {
// ...
private handleBar() {
return this.name + (this.hasBar || '')
}
init = 2;
}
- Unique to TS, you can add modifiers to constructor parameters. This will act as a shorthand that copies them to a class property.
class Foo {
constructor(private name: string) {} // declares a private property "name"
}
While we previously said that TS would avoid adding code-generating features not present in JS, there are some few exceptions from the early versions of the project. This is one of them.
Both worlds
The class
statement differs from most others are it declares both a variable and a type. This is due to the dual nature of JS/OOP classes (a class actually packs 2 objects inside one definition).
class Foo {}
type X = Foo // "Foo - the type" will have the INSTANCE type
type Y = typeof Foo // Y will have the PROTOTYPE type
// (when writing typeof, "Foo" refers to the "living foo",
// which in turn is the prototype)
type Z = InstanceType<Y> // the inverse operation
var foo = new Foo() // "Foo" exists in both worlds;