logo
Orfeas Voutsaridis
profile

Orfeas Voutsaridis

Software Engineer

linkedin
linkedin

Effective TypeScript: Best Practices and Considerations

TypeScript has become an essential tool for modern JavaScript development, offering static typing and other features that enhance code quality and maintainability. This article explores key considerations and best practices for using TypeScript effectively in 2024.

Compiler Options to Keep in Mind

When configuring TypeScript, the tsconfig.json file is crucial. One important option is:

{
    "compilerOptions": {
        "noImplicitAny": true
    }
}

Why noImplicitAny Matters

The noImplicitAny option ensures that all variables have an explicit type, preventing potential runtime errors. Consider the following function:

function add(a, b) {
    return a + b;
}

console.log(add(10, null));

Compiling this with noImplicitAny enabled will result in errors:

index.ts:1:14 - error TS7006: Parameter 'a' implicitly has an 'any' type.
index.ts:1:16 - error TS7006: Parameter 'b' implicitly has an 'any' type.

By enforcing explicit types, you can avoid unexpected behaviors and improve code reliability.

Understanding Code Generation

TypeScript's compiler (tsc) serves two primary functions:

  1. Transpilation: Converts TypeScript to a version of JavaScript that can run in various environments.
  2. Type Checking: Identifies potential errors before runtime, enhancing code safety.

TypeScript and Runtime Performance

TypeScript has no impact on runtime performance. The compiled JavaScript is equivalent to what you would write manually, with no additional overhead.

Limiting the Use of any

TypeScript is designed to be gradual and optional, allowing you to introduce types incrementally. However, overusing any can negate the benefits of TypeScript:

let age: number;
age = "12"; // Error: Type 'string' is not assignable to type 'number'.

age = '12' as any; // Suppresses the error, but at a cost.

While as any can silence errors, it should be used sparingly to maintain type safety.

Refactoring with VS Code

VS Code offers powerful refactoring tools. Use F2 to rename variables or functions across your project, ensuring consistency and reducing errors.

The extends Keyword

The extends keyword is useful in generic constraints, indicating a subset relationship:

function getJet<K extends string>(val: any, key: K) {
    return val[key];
}

getJet({a: 1}, 'a');

Prefer Type Declarations Over Assertions

Type declarations ensure that values conform to expected types, while assertions merely inform the compiler of your intent:

interface Person {
    name: string;
}

const alice: Person = { name: 'Alice' }; // Declaration
const bob = { name: 'Bob' } as Person;   // Assertion

Declarations provide stronger guarantees about your code's correctness.

Avoid Cluttering with Redundant Type Annotations

TypeScript's type inference is powerful. Avoid unnecessary type annotations:

let x: number = 4; // Redundant
let x = 5;         // Preferred

Your editor will display the inferred type, keeping your code clean and readable.

Consistent Variable Types

Ensure variables maintain consistent types to prevent errors:

let id: number = 5;
fetchProduct(id);
id = "23-323"; // Error: Type 'string' is not assignable to type 'number'.
fetchProductSerialNumber(id);

TypeScript will catch these mismatches, helping you maintain type integrity.

New Features in TypeScript 2024

As of 2024, TypeScript continues to evolve with new features and improvements:

  • Variadic Tuple Types: Allow for more flexible function signatures and type manipulations.
  • Satisfies Operator: Provides a way to ensure that a value satisfies a specific type without altering its inferred type.
  • Improved Type Inference: Enhancements in type inference make TypeScript even more powerful and intuitive.

By following these best practices and staying updated with the latest features, you can leverage TypeScript to write more robust, maintainable, and error-free code.

Date: 02/01/2025