Overview
TypeScript 1.8 introduced string literal types for restricting variables to a finite set of possible string values. With TypeScript 2.0, literal types are no longer restricted to string literals. The following literal types have been added to the type system:
In the following sections, we’re going to be looking at a practical example for each of these new literal types.
Boolean Literal Types
The following example defines two constants, TRUE
and FALSE
, which hold the values true
and false
, respectively:
|
|
Trying to assign the opposite boolean value to each of the local variables results in a type error:
|
|
With the introduction of boolean literal types, the predefined boolean
type is now equivalent to the true | false
union type:
|
|
While boolean literal types are rarely useful in isolation, they work great in conjunction with tagged union types and control flow based type analysis. For instance, a generic Result<T>
type that either holds a value of type T
or an error message of type string
can be defined as follows:
|
|
Here’s a function that accepts a parameter .
|
|
Note that with the strictNullChecks
option enabled, string
is a non-nullable type. In order for the function to accept a value of a nullable type for its input
parameter, the null
and undefined
types must explicitly be included in the union type.
We can now call the parseEmailFunction
as follows:
|
|
Here’s a screenshot of Visual Studio Code rendering the above code snippet. Notice that some property access expressions are underlined with red squigglies:
What’s great about this is that the compiler only lets us the value
or error
properties after we’ve checked parsed.success
, our discriminant property:
- If
parsed.success
istrue
,parsed
must have type{ success: true; value: string }
. We can accessvalue
in this case, but noterror
. - If
parsed.success
isfalse
,parsed
must have type{ success: false; error: string }
. We can accesserror
in this case, but notvalue
.
By the way, did you notice that the only TypeScript artifacts in this entire code example are the declaration of Result<T>
and the type annotations in the function signature? The remainder of the code is plain, idiomatic { (语言)自然地道的 } JavaScript that is still fully typed due to control flow based type analysis.
Numeric Literal Types
Similar to string literal types, we can restrict numeric variables to a finite set of known values:
|
|
In practice, we could use a numeric literal when working with port numbers, for example. Unsecured HTTP uses port 80, while HTTPS uses port 443. We can write a getPort
function and encode the only two possible return values in its function signature:
|
|
It gets even more interesting if we combine literal types with TypeScript’s function overloads. That way, we can give more specific types to different overloads of the getPort
function:
|
|
Now, the compiler can help us when it detects conditions that are always return the value false
, for example when comparing httpPort
to the value 443
:
Since httpPort
has type 80
, it always contains the value 80, which of course is never equal to the value 443. In cases like these, the TypeScript compiler can help you detect both buggy logic and dead code.
Enum Literal Types
Finally, we can also use enumerations as literal types. Continuing our example from before, we’ll be implementing a function that maps from a given port (80 or 443) to the corresponding scheme (HTTP or HTTPS, respectively). To do that, we’ll first declare a const enum which models the two port numbers:
|
|
Now comes our getScheme
function, again using function overloads for specialized type annotations:
|
|
Constant enumerations have no runtime manifestation (unless you provide the preserveConstEnums
compiler option) — that is, the constant values of the enum cases will be inlined wherever they are used. Here’s the compiled JavaScript code, with comments removed:
|
|
Super clean, isn’t it?