The following text is a summary based on my understanding after reading "Effective
TypeScript
"
1. Relationship Between TypeScript and JavaScript
TypeScript
is a unique language in terms of its usage. It ultimately compiles into JavaScript
and is executed as JavaScript
.
In terms of its type system, TypeScript
is different from other languages.
💿 You may have often heard the statement, "
TypeScript
is a superset ofJavaScript
with types."
All JavaScript
code is TypeScript
code ➡️ True All TypeScript
code is JavaScript
code ➡️ False
interface City {
name: string;
state: string;
}
const cities: City[] = [
{ name: "Seattle", stete: "Washington" },
{ name: "Los Angeles", state: "California" },
{ name: "Atlanta", state: "Georgia" },
];
In the code above, if there is a typo in the property declaration such as stete, TypeScript
's type checker will catch the error and prevent potential issues that may occur in the future before runtime.
Therefore, TypeScript
programs can be seen as a superset containing both JavaScript
programs and TypeScript
programs that have passed type checking.
Type inference plays a crucial role in TypeScript
, and one of the goals of TypeScript
's type checker (type system) is to find errors in code before runtime. This is why TypeScript
is also referred to as a "static type language."
Thus, we can say that the type system in TypeScript
"models" the runtime behavior of JavaScript
.
2. TypeScript Configuration
Where to find source files? What kinds of output to generate? You can configure TypeScript
compilation settings through the tsconfig.json
file using the compilerOptions
.
// tsconfig.json
{
"compilerOptions": {
"noImplicitAny": true,
"strictNullChecks": false
}
}
When applying TypeScript
to a React
project, it's important to understand the settings for noImplicitAny
and strictNullChecks
.
1. noImplicitAny
const func = (a, b) => a + b;
If noImplicitAny
is set to false
, the type checker will not show any errors. However, if noImplicitAny
is set to true
, the type checker will display the following error:
Parameter a
implicitly has an any
type. Parameter a
implicitly has an any
type.
2. strictNullChecks
const a: number = null;
const b: number = undefined;
If strictNullChecks
is set to false, no errors will be shown. However, if it is set to true
, it will display the error "Type null
is not assignable to type number
" for null
or undefined
values.
If you intentionally want to allow null
, you can use tagged union types (discriminated union types) to prevent errors:
const a: number | null = null;
When strictNullChecks
is true
, you need to use null
checks or type assertions to prevent errors like "undefined
is not an object." For example:
// null check
// 1
if ($elem) {
$elem.textContent = "hello";
}
// 2
render(value);
// type assertion
// 1
$elem!.textContent = "hello";
// 2
render(value as string);
-> In the case of 1, since the DOM can be null
, you can prevent errors by checking for null
or using type assertions. In the case of 2, you can use the "type assertion" syntax, as
, to prevent errors.
Therefore, it is recommended to set the noImplicitAny
option to true
, except when converting a JavaScript
project to TypeScript
. And to prevent runtime errors like "undefined
is not an object," it is recommended to set the strictNullChecks
option to true
.
Additionally, to prevent errors through strict type checking in TypeScript
, it is recommended to set the strict option to true.
3. Understanding Code Generation and Irrelevance of Types at Runtime
The TypeScript compiler performs the following tasks:
Transpiling (translating + compiling) the latest JavaScript
/TypeScript
code to an older version of JavaScript
so that it can run in browsers. Checking for type errors in the code. → These two tasks are completely independent. Transpiling TypeScript
to JavaScript
does not affect the types in the code at all.
💿 This means that code with type errors can still be compiled. It is more accurate to say that if there are type errors, there are errors in type checking.
💿 Type checking is not possible at runtime. In fact, during the compilation process when code is translated into
JavaScript
, all type declaration statements such asinterfaces
andtype aliases
are removed.
This indicates that code generation is independent of runtime behavior and performance. Ultimately, TypeScript
types do not affect runtime behavior or performance.
To use TypeScript
types at runtime, you can use "tagged union types" and "property checking (kind) method," or utilize "classes" to provide both TypeScript
types and runtime values.
1. Tagged union types and property checking (kind) method
interface Animal {
kind: "animal";
name: string;
}
interface Dog extends Animal {
kind: "dog";
name: string;
}
type Content = Animal | Dog;
const callName = (content: Content) => {
if (content.kind === "animal") {
content;
return `this is ${content} type`;
} else if (content.kind === "dog") {
content;
return `this is ${content} type`;
}
};
2. Class + instanceof usage
class Triangle {
constructor(public width: number) {}
}
class IsoscelesTriangle {
constructor(public width: number, public height: number) {
super(width);
}
}
type Shape = Triangle | IsoscelesTriangle;
const calculateArea = (shape: Shape) => {
if (shape instanceof Triangle) {
return (shape.width * shape.height) / 2;
} else if (shape instanceof IsoscelesTriangle) {
return (shape.width * shape.height) / 2;
}
};
Type operations do not affect runtime behavior.
const asNumber = (val: number | string): number => {
return val as number;
};
asNumber(12); // 12
asNumber("12"); // 12
→ The above code passes the type check, but after compilation, the TypeScript
-specific code, such as the as number type assertion, is removed, and only the JavaScript
code remains. Therefore, the type assertion code has no impact at runtime.
Therefore, to refine values based on type definitions, you need to write code like this:
const asNumber = (val: number | string): number => {
return typeof val === "string" ? Number(val) : val;
};
💿 As a result, it is important to be cautious as runtime types and declared types can differ.
4. Getting Familiar with Structural Typing
JavaScript
can be considered a language based on duck typing.
💿 Duck typing is the concept of considering an object to belong to a certain type if it has the variables and methods that conform to that type.
interface Line {
x: number;
y: number;
}
interface NamedLine {
name: "string";
x: number;
y: number;
}
const calculateLength = (l: Line) => {
return Math.sqrt(l.x * l.x + l.y * l.y);
};
const line1 = { x: 1, y: 2 };
const line2: NamedLine = { x: 3, y: 4, name: "z" };
calculateLength(line1); // 2.23606797749979
calculateLength(line2); // 5
In the above code, even though the object has the NamedLine interface, it can be passed as an argument to calculateLength
without any issues. TypeScript
, even during type checking, is intelligent enough to recognize that the object has properties x
and y
even though they are not explicitly defined in the Line
interface. This is known as structural typing.
💿 While writing functions, it is often assumed that the arguments used in function calls adhere strictly to the properties defined in the function's parameter type. However, in
TypeScript
, types have open properties instead of sealed properties.
5. Avoiding the use of any Type
Using the any type undermines the type checking system provided by TypeScript
and makes it difficult to handle various runtime errors that may occur during JavaScript
execution after TypeScript
compilation.
Moreover, when working collaboratively, it becomes challenging to understand the design structure of a service when any types are used extensively. Therefore, it is advisable to avoid using the any type and instead specify clear and explicit types.
The above summary is based on my understanding while reading the book. If there are any additional concepts you would like me to explain, please let me know.