Skip to content

Latest commit

 

History

History
80 lines (77 loc) · 9.64 KB

README.md

File metadata and controls

80 lines (77 loc) · 9.64 KB

🧩 Patterns for JavaScript, TypeScript, Node.js

Rethinking GRASP (General Responsibility Assignment Software Patterns), SOLID (Single responsibility, Open–closed, Liskov substitution, Interface segregation, Dependency inversion), GoF (Gang of Four) patterns, for Frontend (browsers) & Backend (node.js, other runtimes) development with JavaScript and TypeScript

Translations: EN, UA, RU.

  • 🧩 Patterns
    • 📢 GoF patterns for Node.js and JavaScript (seminar fragment)
    • 🏭 Creational patterns
      • Abstract factory — creates related objects belonging to one family without specifying their concrete classes, e.g., UI components for different platforms.
      • Builder — step-by-step assembly of a complex configurable object, often using chaining, e.g., Query Builder or Form Generator.
      • Factory — function or method that creates objects using different techniques: assembling from literals and methods, mixins, setPrototypeOf.
      • Factory Method — chooses the correct abstraction to create an instance; in JavaScript, this can be implemented using if, switch, or selection from a collection (dictionary).
      • Prototype — creates objects by cloning a prepared instance to save resources (not to be confused with Prototype-programming, which is closer to Flyweight).
      • Flyweight — saves memory allocation by sharing common state among multiple instances.
      • Singleton — provides global access to a single instance; often considered an anti-pattern, easiest implemented via ESM/CJS module caching exported refs.
      • Object Pool — reuses pre-created objects to save resources during frequent creation and destruction.
    • 🤝 Structural patterns
      • Adapter — converts an incompatible interface into a compatible one, enabling third-party component usage without altering its code; can even transform a function contract into an object or vice versa.
      • Wrapper — function wrapper that delegates calls and adds behavior; a specialized case of Adapter.
      • Boxing — wraps primitives into object types to add methods or unify interfaces, e.g., narrowing String to AddressString.
      • Decorator
      • Decorator — dynamically extends behavior without inheritance, typically via composition and declarative syntax, effectively adding metadata.
      • Proxy — controls access to an object by intercepting calls, reads, and writes; useful for lazy initialization, caching, and security; can be implemented via GoF or native JavaScript Proxy.
      • Bridge — separates two or more abstraction hierarchies via composition or aggregation, allowing them to evolve independently.
      • Composite — implements a common interface to uniformly handle individual objects and their tree structures, e.g., DOM or file systems.
      • Facade — simplifies access to a complex system, providing a unified and clear interface, hiding and protecting internal complexity.
      • Flyweight — saves memory allocation by sharing common state among multiple instances.
      • Active Record — domain object encapsulating a database record, providing methods to directly perform CRUD operations (create, read, update, delete) and domain-specific queries on itself.
      • Data access object (DAO) — abstraction defining an interface to persist and retrieve domain objects, isolating domain logic from specific storage implementations.
      • Data transfer object (DTO) — an anemic object (just plain data) carrier without domain behavior, designed explicitly for transferring structured data across application boundaries, layers, modules, or subsystems.
      • Data Access Layer (DAL) — a layer abstracting access to multiple DAOs or raw data sources. Can be represented as Facade pattern. Often includes transformations.
      • Repository — domain-centric abstraction for data access that returns domain entities, not raw data or DTOs.
    • ⚡ Behavioral patterns
      • Chain of Responsibility — passes control through a chain of handlers, selecting a responsible one; all handlers can read, but only one will modify.
      • Middleware — handler chain similar to CoR, but each can modify state and pass control to the next one, potentially leading to race conditions and conflicts.
      • Command — encapsulates an action (execution request) and parameters into an object, allowing queuing, cancellation, repetition, etc.
      • Interpreter — implements a DSL language (Domain Specific Language) or parses expressions into AST (Abstract Syntax Tree) for interpretation.
      • Iterator — sequentially traverses collections or streams element-by-element without exposing all data; JavaScript provides built-in Iterator and AsyncIterator.
      • Mediator — optimizes communication between N components, centralizing interaction to reduce coupling from N*(N-1)/2 down to N.
      • Memento — saves and restores snapshots of an object's state without direct access to its internal state.
      • Observable — notifies subscribers about changes to an object's state via Events:
      • State — implements a Finite State Machine (FSM) where methods represent transitions, and state is composed into abstraction and switched during transitions.
      • Strategy — selects interchangeable behavior at runtime from a collection of implementations: functions, objects, or classes
      • Template method — defines algorithm steps, allowing subclasses to override individual steps while defaulting to the superclass behavior.
      • Visitor — adds operations to objects without altering their classes, separating structure and behavior into distinct abstractions.
      • Revealing Constructor — changes behavior without inheritance, injecting functionality into constructors via functions or objects describing the behavior.
      • Actor – Encapsulates state and behavior, communicating asynchronously via message passing and processing messages in a queue. Ensures thread-safe and async-safe concurrent operations by isolating actor state.
      • Reactor (event-loop) - Handles concurrent events synchronously by adding them to queue and dispatching them to registered handlers. Implements event-driven async processing on the top of the sync one; commonly used in I/O-bound systems.
      • Proactor - Event loop where operations started by user-land code but completed by an external agent (for example I/O subsystem), which then triggers a completion handler when the operation finishes (returning data to callback).
      • Service Locator —
  • 🧩 GRASP patterns
  • 🧩 SOLID Patterns