TypeScript Strict Mode — Missing Null Check Broke Payments
Orders completed with zero amounts — a missing null check on user.phoneNumber caused payment failures.
20+ years shipping production JavaScript and front-end systems at scale. Notes here come from systems that actually shipped.
- TypeScript adds static types to JavaScript — catches bugs at write-time, not runtime
- Interface describes object shapes; type alias handles unions and intersections
- Strict mode (strict: true) is non-negotiable for production safety
- Generics give you reusable, type-safe abstractions without sacrificing flexibility
- The biggest mistake: using
anyas a crutch — it disables the type system and invites runtime failures
Imagine you're building with LEGO. Plain JavaScript is like a box of mixed bricks with no labels — you can grab any piece and try to snap it anywhere, only discovering it doesn't fit after you've built half the castle. TypeScript is like a labelled LEGO kit: every bag is marked, every piece has a designated slot, and the instructions tell you before you even start if you're trying to put a wheel where a window should go. The 'mistake' is caught before the castle collapses.
JavaScript runs half the internet, but it has a well-known Achilles heel: it will happily let you pass a number where a string is expected, call a method on undefined, or silently return NaN and keep going. These bugs don't blow up at write-time — they blow up at 2am in production, after a user has already seen broken behaviour. That's not a flaw in your skill; it's a flaw in the language's design philosophy. JavaScript was built for small scripts, not 100,000-line enterprise applications.
TypeScript is Microsoft's answer to that problem. It's a strict superset of JavaScript — meaning every valid JS file is already valid TypeScript — that adds a static type system on top. The TypeScript compiler reads your types, checks your logic against them, then compiles everything down to plain JavaScript that any browser or Node.js process can run. The types vanish at runtime; they exist purely to protect you at development time. Think of them as scaffolding: essential while you're building, removed once the structure is sound.
By the end of this article you'll understand not just what TypeScript syntax looks like, but why the type system is designed the way it is, how to use interfaces and generics to model real-world data structures, and which mistakes will waste your time if you don't know about them upfront. You'll walk away ready to migrate a JavaScript project to TypeScript or start a new one with confidence.
What TypeScript Strict Mode Actually Enforces
TypeScript strict mode is a compiler flag that enables a set of type-checking rules designed to catch null safety and implicit any violations at compile time. It activates strictNullChecks, noImplicitAny, strictFunctionTypes, and others, turning the type system from a loose annotation layer into a sound null-safety guard. Without it, null and undefined are assignable to any type, which silently permits runtime null pointer exceptions in production.
When strict mode is on, every variable, parameter, and return type must explicitly account for null or undefined using union types like string | null. This forces developers to handle missing values at every boundary — function calls, API responses, database queries. The compiler rejects code paths where a nullable value is used without a check, effectively making null safety a compile-time guarantee rather than a runtime hope.
Use strict mode in every TypeScript project from day one. In payment systems, a missing null check on a transaction amount or user ID can cause silent failures, charge errors, or data corruption. Strict mode eliminates entire categories of bugs that slip into production, especially in data-intensive flows where null values are common. It's not optional — it's the baseline for any system that processes money or user data.
transactionId from a failed DB query was passed to the refund API, causing a double charge.// @ts-strict comments.Static Typing vs Dynamic Typing — Why TypeScript Exists at All
In JavaScript every variable is dynamically typed — its type is determined at runtime based on the value it holds at that moment. That flexibility is useful for tiny scripts. But in a large codebase, it means you're constantly holding a mental model of what every variable 'should' be, because the language won't enforce it for you. Functions accept any argument, return values are ambiguous, and refactors become terrifying guessing games.
TypeScript introduces static typing: you declare what type a value is when you write the code, and the compiler verifies every usage before the code ever runs. If you write a function that expects a User object and you accidentally pass a plain string, TypeScript tells you immediately — right in your editor, with a red squiggle and a precise error message.
This isn't just about catching typos. It's about making your code self-documenting. A function signature like function sendEmail(recipient: User, subject: string): Promise<void> tells every developer on your team exactly what it needs and what it returns, with zero ambiguity. No more digging through documentation or console-logging to figure out what shape a value is. The types are the documentation.
.d.ts declaration file — avoid using any or the library will be untypedInterfaces and Type Aliases — Modelling Real-World Data Structures
Once you have more than a handful of typed variables, you need a way to describe the shape of complex objects — things like API responses, database records, or configuration objects. TypeScript gives you two tools for this: interface and type. They look similar and are often interchangeable, but they have meaningful differences worth understanding.
An interface is specifically designed to describe the shape of an object. It can be extended (like class inheritance), merged across multiple declarations, and implemented by a class. A type alias is more general — it can represent primitives, unions, intersections, tuples, and objects. If you're modelling an object that other parts of your code will implement or extend, reach for interface. If you need a union type or something more compositional, reach for type.
In a real project you'll use both. An API response might come back as User | null — that's a union, so type is appropriate. The User shape itself is a contract that components and services must honour — that's interface territory. Understanding when each tool is the right fit is what separates TypeScript beginners from developers who write genuinely maintainable typed code.
interface for object shapes that describe a 'thing' in your domain (User, Product, Order). Use type when you need unions (A | B), intersections (A & B), or to alias a primitive. When in doubt, interface is more extensible and gives better error messages.Interface vs Type: Detailed Comparison Matrix
While both interface and type can define object shapes, they have distinct capabilities and limitations that affect how you model data. Choosing the wrong one can lead to verbose code or unexpected behavior when refactoring. Here's a side-by-side comparison of their key features:
| Feature | Interface | Type Alias | ||
|---|---|---|---|---|
| Primitive alias | ❌ Cannot alias string, number | ✅ Can alias primitives, e.g., type ID = string | ||
| Union types | ❌ Cannot represent `A | B` | ✅ `type Result = Success | Failure` |
| Intersection types | Can use extends for composition | ✅ Using & operator | ||
| Declaration merging | ✅ Multiple declarations merge | ❌ Duplicate type redeclaration error | ||
| Computed/mapped types | ❌ Limited support | ✅ Full support for keyof, Record<K,T>, Pick<T,K> | ||
| Error messages | Clearer – conflict shown at declaration | Sometimes cryptic – conflict shown at usage | ||
| Extending other types | extends keyword | Intersection & (can lead to silent overrides) | ||
| Class implementation | ✅ implements InterfaceName | ✅ Also works, but less conventional | ||
| Performance | Slightly faster for large types (cached) | Slightly slower for deeply nested intersections |
This table isn't just academic: choosing the wrong tool directly impacts compile-time error quality and maintainability. For production code where object shapes serve as shared contracts (e.g., API request/response models, database entities, component props), always start with interface. Reserve type for scenarios that demand unions, computed types, or primitive aliases. When you need both extension and union, consider splitting: define the base shape as an interface, then create union types from it.
interface. If you need to combine or transform types (unions, intersections, computed) → use type. When in doubt, start with interface and switch to type only when you hit a limitation.type intersections for large object shapes, a property conflict may compile silently — the last type wins — and you won't discover the bug until runtime. With interface extends, TypeScript reports the conflict immediately at the declaration. This alone is a strong reason to prefer interface for core domain models.Generics — Writing Code That Works for Any Type Without Losing Safety
Here's a problem you hit quickly in TypeScript: you want to write a reusable utility function — say, one that wraps an API response — but you don't want to lose type safety by falling back to any. If you type the parameter as any, TypeScript stops checking it and you've thrown away all the benefits you came here for.
Generics solve this elegantly. A generic is a type placeholder — a variable for a type rather than a value. You write the function once with a generic parameter like <T>, and TypeScript figures out what T actually is at each call site based on what you pass in. You get full type safety AND full reusability. It's one of the features that genuinely changes how you think about writing abstractions.
Generics show up everywhere in real TypeScript codebases: API client wrappers, data transformation utilities, React component props, custom hooks, and state management patterns. Once you're comfortable with them, you'll find yourself reaching for them constantly instead of duplicating code or compromising on any.
response.json() as T, TypeScript believes you — it doesn't actually validate that the JSON matches T at runtime. If your API returns a different shape than expected, TypeScript won't catch it. For production apps, use a runtime validation library like Zod alongside TypeScript to validate API responses before you use them.any inside a generic function defeats the purpose — you lose all type safety.any in a generic constraint; prefer unknown and narrow the type explicitly.Common Utility Types Cheat Sheet: Pick, Omit, Partial
TypeScript ships with several built-in utility types that save you from manually rewriting object shapes. The most frequently used in production code are Partial, Pick, and Omit. These are mapped types: they transform an existing type into a new type by adding or removing optionality or selecting specific keys. Understanding them is essential because they let you create derived types without duplicating interfaces, keeping your type definitions dry.
Partial<T> — makes every property in T optional. Use when you need to represent a partial update (e.g., PATCH request body) or a partially constructed object. Be careful: accessing a property that may now be undefined requires null checks.
Pick<T, K> — creates a type with only the specified keys K from T. Use when you want a subset of an interface, like a lightweight view of a user record that only includes name and email.
Omit<T, K> — creates a type with all keys from T except the specified keys K. Use when you want to exclude sensitive fields (e.g., password, ssn) from a response type, or when you need to remove a key before processing.
These utilities are the workhorses of real-world TypeScript projects. Combined with generics, they let you build type-safe data transformers, update functions, and API response filters with minimal boilerplate.
Partial<T> accepts {}. If you iterate over keys of a Partial, you may end up with no operations. Always validate that at least one expected property was provided before performing an update. Consider using a union of partials with a discriminant or a custom type like AtLeastOne<T> if you need guarantees.Pick and Omit to derive response types from domain models is a common pattern for preventing exposure of sensitive fields. However, nesting utilities (e.g., Partial<Pick<T, K>>) can make error messages harder to read. Keep utility chains short and document the resulting shape for maintainability.tsconfig.json — The Settings That Make TypeScript Actually Strict
Installing TypeScript and adding a few type annotations doesn't automatically make your code safe. TypeScript has a spectrum of strictness controlled by tsconfig.json, and the default settings are surprisingly permissive. The most important flag is strict: true — it's actually a shorthand that enables a bundle of individual checks, the most consequential being strictNullChecks.
Without strictNullChecks, TypeScript allows null and undefined to be assigned to any type. That means a variable typed as string could actually be null at runtime, and TypeScript won't complain. With it enabled, null and undefined become their own distinct types. A function that might return nothing must say so explicitly with a return type like string | null, and every caller must handle that null case. This is uncomfortable at first and absolutely worth it.
Other key flags: noImplicitAny prevents TypeScript from silently falling back to any when it can't infer a type — you're forced to be explicit. noUnusedLocals keeps your codebase clean. target and module control what JavaScript version TypeScript compiles down to. Starting a new project? Set strict: true from day one. Migrating an existing project? Enable flags incrementally to avoid being overwhelmed.
strict: false, add noImplicitAny: true alone, and fix those errors first. Then add strictNullChecks: true and fix those. One flag at a time makes the migration manageable and gives your team concrete progress milestones.Essential tsconfig.json Flags: strict, noImplicitAny, and More
While the previous section explained the strict mode philosophy, this is your quick reference for the most important compiler flags you need to know for production TypeScript. Every flag here has a direct impact on the safety and maintainability of your codebase.
| Flag | Effect | Why It Matters |
|---|---|---|
strict: true | Enables all strict type-checking options. | Without it, TypeScript loses ~70% of its safety. This is a one-line kill switch for the entire strict family. |
strictNullChecks | null and undefined become distinct types. | The #1 cause of production bugs in TypeScript projects that disable it. Every variable that can be null must be handled. |
noImplicitAny | Errors when TypeScript cannot infer a type and silently falls back to any. | Prevents accidental escape hatches. Without it, a variable typed as any disables type checking for that entire chain. |
strictFunctionTypes | Checks function parameter types more rigorously (contravariance). | Catches unsafe subclassing where a callback expects a more specific type than provided. |
noUnusedLocals | Errors on unused local variables. | Keeps dead code out of your codebase, especially important during refactors. |
noUnusedParameters | Errors on unused function parameters. | Prevents misleading function signatures. Use prefix underscore (_) to intentionally ignore. |
noImplicitReturns | Errors when a function has code paths that don't return a value. | Prevents accidental undefined returns when you intended to always return a value. |
exactOptionalPropertyTypes | Ensures optional properties don't allow undefined implicitly. | Prevents a common pitfall where { name?: string } could be set to undefined explicitly. |
For new projects, set strict: true and never look back. For migrations, start with noImplicitAny and strictNullChecks. Once those compile clean, turn on strictFunctionTypes. The table above serves as a checklist you can paste into your pull request template as a config review reminder.
noImplicitAny and fix all errors (may need any in some places, but note where). Week 2 – enable strictNullChecks and start using optional chaining. Week 3 – enable strictFunctionTypes and refactor any callbacks that fail. Week 4 – flip strict: true and handle remaining edge cases.strict: false is a potential production crash. Invest the time upfront — it's cheaper than a post-mortem at 2 AM.strict: true is the baseline; add noUnusedLocals, noUnusedParameters, and noImplicitReturns for an additional layer of tidiness.Type Narrowing and Type Guards — Writing Code That Handles Multiple Types Safely
Once you start using union types (e.g., string | null, Success | Failure), you'll hit a wall: TypeScript won't let you access properties or call methods that only exist on one branch of the union. You need to narrow the type before you can use it. Type narrowing is the process of refining a broad type into a more specific one based on runtime checks.
TypeScript understands common JavaScript patterns as type guards: typeof, instanceof, in, equality checks, and user-defined predicates. When you write if (typeof value === 'string'), TypeScript knows that inside that if block, value is a string — and it can safely let you call .toLowerCase(). Outside the if, it's still the union type.
User-defined type guards are a powerful pattern for complex checks. A function that returns a predicate like value is string tells TypeScript that when the function returns true, the argument is that specific type. This is invaluable for validating API responses, discriminating unions, and parsing untrusted data.
kind or status (e.g., { kind: 'success', data: T } | { kind: 'error', error: string }). Then you can narrow with a simple if (response.kind === 'success') — it's cleaner and more maintainable.Why Your Codebase Is Still Full of `any` — And How to Kill It for Good
You know the drill. Some junior—or let's be honest, probably you at 2 AM—slaps as any on a response from an API and calls it a day. That any doesn't just turn off type checking. It erases the entire contract between your code and the data it processes. When that API shape changes next sprint, you don't get a compile-time error. You get a Cannot read properties of undefined in production at 3 PM on a Friday.
The fix isn't just 'use strict mode.' You need a zero-tolerance policy for implicit any in function returns. TypeScript's noImplicitReturns flag catches functions that forget to return a value. But the real weapon is strictNullChecks combined with noUncheckedIndexedAccess. That combo forces you to handle every undefined branch explicitly. No more optional chaining as a crutch—you actually check the thing exists before you touch it.
Here's the pattern that senior teams enforce: every function has an explicit return type. If you see a function without one, that's a code review fail. Period. When you type the return, the compiler catches mistakes you didn't even know you made. And when you're consuming external data, use a runtime validator like Zod or io-ts to bridge the gap between the wild west of JSON and your typed world. TypeScript doesn't run at runtime—your validators do.
as casts on API responses is the number one source of runtime crashes in typed TypeScript codebases. The compiler trusts you. Don't make it a liar.The Union Type Trick That Prevents Invalid States at Compile Time
Most devs think union types are just for 'string or number.' That's like using a scalpel to open a can of beans. The real power of union types is modelling mutually exclusive states so the compiler eliminates entire classes of bugs before they compile.
Consider a form that can be in three states: idle, submitting, or error. The naive approach is three booleans: isIdle, isSubmitting, isError. But booleans are a liar's data type—they let you represent states that don't make sense, like isIdle=true and isSubmitting=true simultaneously. That's a bug that lives in the gap between what you intend and what the type system allows.
Instead, model the state as a discriminated union. Each variant carries only the data relevant to that state. When you're in 'submitting', there's no error message field. When you're in 'error', there's no result data. The compiler enforces this—not your code review, not your tests, not your prayers. This pattern is called 'making illegal states unrepresentable.' It's not buzzword fluff—it's the difference between a codebase you trust and one you're scared to deploy on a Friday.
Apply this to API responses, UI states, navigation stacks—anywhere you'd reach for an enum or a set of flags. Your type definitions become executable documentation that the compiler enforces.
Classes Are Not Your Dump Truck — When OOP Actually Pays Off in TypeScript
TypeScript classes aren't Java. You don't need a class for every noun in your domain model. The real value of classes in TypeScript is encapsulation of state with guaranteed invariants, not inheritance pyramids that collapse in month two.
Use classes when you have mutable state that must stay consistent — a bank account that can't go negative, a WebSocket connection manager that enforces a max reconnection count. For data-only structures, interfaces or type aliases are cheaper and easier to reason about. The private, protected, and readonly modifiers exist so you can enforce constraints from the constructor onward, not so you can write getters for every field.
The implements keyword is your friend. Code against interfaces, not concrete classes. That way you can swap implementations without touching callers. And please, stop putting business logic in constructors. Build objects with factory functions or static methods, keep constructors simple assignments.
Testing TypeScript Means Testing The Types, Not Just The Values
Your test suite should fail when types break before a single runtime assertion runs. Unit tests prove your logic works for given inputs. Type tests prove your generics, unions, and overloads actually constrain callers correctly.
Use @ts-expect-error in tests — place it on lines you expect to be type errors. If the line compiles without error, the test fails. This catches API contract violations during CI. For example, if you change a function signature, the type test that passed a string where a number was expected will now fail to compile, and your type-level test pinpoints exactly which callers broke.
For runtime testing, pick Vitest over Jest. It's faster, uses the same describe/it/expect API you already know, and handles TypeScript and ESM natively without babel acrobatics. Mock minimally — if your module is hard to test without mocking, your module is coupled wrong, not your test framework.
Production rule: a PR that changes types without adding a type test is incomplete. Period.
expectTypeOf for even cleaner assertions.@ts-expect-error to assert type errors — your CI should catch contract breaks.Classes Aren’t a Silver Bullet — When OOP Actually Pays Off in TypeScript
TypeScript classes exist for when you need encapsulated state that changes over time — a bank account, a game character, a WebSocket connection. If your data is just a blob of fields, use an interface or type alias. Classes add runtime cost, constructor boilerplate, and prototypal inheritance complexity. Reach for a class when you need methods that mutate internal state, when you want to enforce invariants via private fields, or when you rely on instanceof type guards. For pure data transfer objects, interfaces with factory functions are lighter and compose better. The trap: using classes everywhere because you came from Java or C#. That kills TypeScript’s structural typing advantages. Reserve classes for objects with behavior that must stay coupled to mutable state. Everything else stays plain objects with functions.
TypeScript Interview Questions That Test Real Understanding
Interviewers love asking about keyof, conditional types, and the infer keyword — because those reveal whether you truly grasp TypeScript’s type system or just skate by with any. Expect: "How does typeof differ in TypeScript vs JavaScript?" (JS: runtime string tag; TS: compile-time type query). "What does extends do in generics vs classes?" (Generics: constraint; Classes: inheritance). "Write a type that extracts the return type from a function." (Use infer R in a conditional type). The real test: explain why Pick<T, K> is safer than Omit<T, K> when K is a union (Pick fails at compile-time if K has an invalid member; Omit silently returns T). Another classic: "How do discriminated unions prevent invalid states?" Show a union of objects with a kind literal field, then switch on it — TypeScript narrows the payload per case. Master these patterns, and you’ll pass any senior-level TypeScript screen.
ReturnType — it breaks on overloaded functions. Use Awaited<ReturnType<...>> for async.The Missing Null Check That Crashed Payment Processing
processPayment function had a user.phoneNumber access without a null check. With strictNullChecks disabled, phoneNumber could be undefined, but TypeScript treated it as string. The subsequent toUpperCase() call returned undefined, and the payment gateway received an invalid customer ID, silently failing.strict: true in tsconfig.json. This immediately surfaced 47 null-safety errors. Added explicit null checks and used optional chaining (user?.phoneNumber?.toUpperCase()) and default values. The payment function now returns a typed union PaymentResult | PaymentError.- TypeScript without strictNullChecks gives you ~30% of the safety you paid for. Always enable strict: true from day one.
- A clean compile does not mean correct runtime behavior — types are only as good as the constraints you enforce.
- Use code reviews to check for use of
anyand missing null handling. Automate this with ESLint rules.
tsc --noEmit to get exact line. Use typeof or keyof to inspect types in your editor. If the property exists at runtime but not in type, consider using a type assertion or updating the interface.noImplicitAny in tsconfig. Use generics if the type is unknown but should be constrained. Avoid any - prefer unknown and type narrowing.?.), nullish coalescing (??), or explicit checks (if (value !== undefined)) to narrow the type before assignment.tsc --noEmit --prettyIn your editor, hover over the object literal to see expected typeKey takeaways
compiler flag is not optional for serious projects — without it (especially without strictNullChecks`) you're getting maybe 30% of TypeScript's actual safety benefits.any when you need flexibility without sacrificing correctness.interface is a contract for an object's shape and is the right default for domain models. A type alias is more powerful for unions, intersections, and compositional patternsCommon mistakes to avoid
4 patternsUsing `any` as an escape hatch everywhere
any with unknown when the type is genuinely unknown. With unknown TypeScript forces you to narrow the type before using it (e.g., if (typeof value === 'string')) rather than silently trusting you. Reserve any only for truly unavoidable third-party interop, and enable the noImplicitAny compiler flag so TypeScript flags accidental any usages.Conflating `interface` extending with `type` intersections and getting confused by error messages
& expecting them to behave like an interface extension, but conflict errors are cryptic and hard to debug.interface and extends. Use & intersections for mixing in utility shapes like { createdAt: Date } onto an existing interface.Forgetting that TypeScript types are erased at runtime
if (myValue instanceof MyInterface) expecting it to work, but you get a runtime error because interfaces don't exist in compiled JavaScript.typeof, instanceof for classes, or custom guard functions with value is MyType return types), or use a schema validation library like Zod which generates both the TypeScript type AND a runtime validator from a single schema definition.Not enabling strictNullChecks in production projects
strict: true in tsconfig.json. This enables strictNullChecks along with other strict flags. Add explicit null checks everywhere. Use optional chaining and nullish coalescing to handle undefined paths gracefully.Interview Questions on This Topic
What is the difference between `interface` and `type` in TypeScript, and when would you choose one over the other?
interface is designed for extensibility (supports declaration merging and extends) while type is more general (can represent unions, intersections, tuples). Use interface for domain models that other code will inherit or implement. Use type for unions (A | B), intersections (A & B), or when you need a computed type. Interfaces give better error messages when merging conflicts occur.Frequently Asked Questions
20+ years shipping production JavaScript and front-end systems at scale. Notes here come from systems that actually shipped.
That's TypeScript. Mark it forged?
15 min read · try the examples if you haven't