design-pattern

Applies object-oriented design principles and design patterns to generate maintainable, extensible code. Use when generating code that requires proper architectural layering, SOLID principles, and appropriate design patterns to solve recurring software design problems.

Safety Notice

This listing is imported from skills.sh public index metadata. Review upstream SKILL.md and repository scripts before running.

Copy this and send it to your AI assistant to learn

Install skill "design-pattern" with this command: npx skills add cylixlee/skills/cylixlee-skills-design-pattern

Design Pattern

This skill guides the generation of high-quality, maintainable, and extensible code by applying established software design principles. When generating code for any non-trivial application, always follow these principles to ensure the resulting codebase is easy to understand, modify, test, and extend over time.

Core Principles

1. Simplicity First

Always prefer simple solutions over complex ones. Do not over-engineer or add abstraction layers that are not yet needed.

YAGNI (You Aren't Gonna Need It)

Do not design abstractions "just in case" they might be needed later. Implement what is needed now, and refactor when actual extension requirements arise. Premature abstraction often leads to incorrect interfaces that must be changed anyway.

KISS (Keep It Simple, Stupid)

Simple code is easier to understand, test, and maintain. If a simple solution works, use it. Complexity should be added only when there is a clear, immediate need.

2. Interface Design

Prefer Standard Library Interfaces

When the standard library or language provides an interface that meets your needs, use it. Do not create custom interfaces that duplicate standard ones. For example, in Go use io.Reader and io.Writer instead of defining your own Reader or Writer interfaces.

Define Interfaces at the Point of Need

Define interfaces where they are consumed, not at a central location. This reduces unnecessary coupling and allows interfaces to be specific to actual use cases.

3. Package and Module Organization

Name Packages by Purpose, Not Project Name

Name packages based on what they do, not the project name. For example, a parser package should be called "parser" or "parsing", not "brainvm".

Follow Language Conventions

Use the project structure and conventions appropriate for the language:

  • Go: Single program → root main.go, multiple programs → cmd/ folder, library → functional packages
  • Java: Standard Maven/Gradle structure with src/main/java
  • Python: src/ or flat structure with __init__.py
  • Do not assume all languages have the same conventions (not all languages have cmd/)

4. Layered Architecture

Layered Architecture is essential for maintainable software because it separates concerns, making code easier to understand, test, and modify. When all code is mixed together in a single layer, changes in one area can unexpectedly break unrelated functionality. By separating responsibilities into distinct layers, each layer can be developed, tested, and modified independently.

The key principle is that dependencies should only flow in one direction, typically from outer layers toward inner layers. This ensures that core business logic does not depend on external concerns like databases, UI frameworks, or external services.

Why Layered Architecture Matters

  • Separation of Concerns: Each layer has a single, well-defined responsibility
  • Testability: Inner layers can be tested without involving outer layers
  • Maintainability: Changes to one layer (e.g., switching databases) don't affect other layers
  • Reusability: Core business logic can be reused across different interfaces
  • Parallel Development: Different teams can work on different layers simultaneously

Common Layering Approaches

Model-View-Controller (MVC): A pattern that separates an application into three main components: the Model (data and business logic), the View (presentation layer), and the Controller (handles user input and orchestrates Model and View). MVC is particularly common in web applications and UI frameworks.

Domain-Driven Design (DDD): DDD emphasizes modeling software around business domains. It uses concepts like entities, value objects, aggregates, bounded contexts, and domain services. DDD is particularly valuable for complex business applications where the domain logic is sophisticated and frequently changing. For more details, see Domain-Driven Design.

General Rule

Regardless of the specific layering approach chosen, always ensure:

  • Each layer has clear, single responsibility
  • Dependencies flow inward only
  • Layer boundaries are enforced through interfaces or abstractions
  • Code is distributed across multiple files and packages, not lumped together

Avoid Unnecessary Middle Layers

Do not create abstraction layers that merely wrap other layers without adding value. Only create a new layer when there is a clear separation of concerns that genuinely needs it. If two components naturally belong together, keep them together.

2. OOP Over Imperative

Write code using Object-Oriented Programming principles rather than procedural or functional style. OOP provides better abstraction, encapsulation, and polymorphism for building complex systems.

Use Interfaces or Abstract Classes to Define Contracts

Always define the "what" separately from the "how". Define what operations are available through an interface or abstract class, then provide concrete implementations. This allows clients to depend on abstractions rather than specific implementations, enabling flexibility and testability.

Prefer Composition Over Inheritance

Favor "has-a" relationships over "is-a" relationships. Inheritance creates tight coupling between classes, making changes difficult and testing problematic. Composition (组装) allows you to combine behaviors by injecting dependencies, which is more flexible and easier to test.

Encapsulate Behavior Within Classes

Keep related behavior and data together. Avoid scattered utility functions or global state. Each class should bundle its data with the operations that manipulate that data.

Extract Hardcoded Logic to Strategy Classes

When you find conditional logic that varies by type or category, consider extracting each variant into its own class. This follows the Strategy pattern and makes the code easier to extend without modification.

3. SOLID Principles

Apply these five principles as code quality checks. Whenever you write code, verify that it does not violate any SOLID principle.

Single Responsibility Principle

Every class should have only one reason to change. A class should do one thing well. If a class is responsible for multiple concerns, changes to one concern may affect others unexpectedly.

Open/Closed Principle

Software entities should be open for extension but closed for modification. Add new features by adding new code rather than modifying existing code. This is typically achieved through abstraction, polymorphism, and composition.

Liskov Substitution Principle

Subtypes must be substitutable for their base types. A subclass should honor the contract of its parent class. If a subclass cannot fully implement parent behavior, the inheritance relationship is incorrect.

Interface Segregation Principle

Clients should not be forced to depend on interfaces they do not use. Many small, specific interfaces are better than one large interface. This prevents classes from being burdened with methods they don't need.

Dependency Inversion Principle

High-level modules should not depend on low-level modules. Both should depend on abstractions. This enables swapping implementations without affecting clients.

4. Design Pattern Selection

Design patterns are reusable solutions to commonly occurring problems. Choose appropriate patterns based on the specific problem context.

How to Use Design Pattern References

  1. Analyze the problem: First, understand what problem you need to solve
  2. Select the pattern: Choose the appropriate design pattern from the tables below based on the problem type
  3. Choose implementation: Navigate to references/{pattern-name}/{language}.md for your target language
  4. Apply with care: Remember that patterns are tools, not rules - use them when they fit

Creation Patterns

PatternPurposeGoJavaPython
Factory MethodDelegate instantiation to subclassesGoJavaPython
Abstract FactoryCreate families of related objectsGoJavaPython
BuilderConstruct complex objects step by stepGoJavaPython
SingletonEnsure single instance with global accessGoJavaPython
PrototypeCreate objects by cloning existing onesGoJavaPython

Structural Patterns

PatternPurposeGoJavaPython
AdapterConvert interface compatibilityGoJavaPython
BridgeSeparate abstraction from implementationGoJavaPython
CompositeTree structures for part-whole hierarchiesGoJavaPython
DecoratorAdd responsibilities dynamicallyGoJavaPython
FacadeSimplified interface to complex subsystemGoJavaPython
FlyweightShare common state efficientlyGoJavaPython
ProxyControl access to another objectGoJavaPython

Behavioral Patterns

PatternPurposeGoJavaPython
Chain of ResponsibilityPass request along a chain of handlersGoJavaPython
CommandEncapsulate request as an objectGoJavaPython
IteratorAccess elements sequentiallyGoJavaPython
MediatorCentralized communicationGoJavaPython
MementoCapture and restore internal stateGoJavaPython
ObserverNotify dependents of state changesGoJavaPython
StateAlter behavior based on internal stateGoJavaPython
StrategyInterchangeable algorithmsGoJavaPython
Template MethodDefine algorithm skeleton, let subclasses override stepsGoJavaPython
VisitorOperations on elements of an object structureGoJavaPython

5. Dependency Management

Proper dependency management is critical for maintainability and testability.

Use Dependency Injection

Inject dependencies rather than creating them inside classes. Pass dependencies through constructors or setters rather than having classes create their own dependencies. This enables loose coupling and makes testing straightforward.

Inject Abstractions, Not Concretions

Always depend on interfaces or abstract types, not concrete implementations. This allows implementations to be swapped without affecting client code.

Avoid Circular Dependencies

Circular dependencies make code hard to test and reason about. The dependency graph should be acyclic. If two modules depend on each other, extract shared abstractions to break the cycle.

6. Testability

Design code to be easily testable. Testable code is usually well-designed code.

  • Avoid static state that persists between tests
  • Avoid singletons with mutable state
  • Use dependency injection to enable mocking
  • Favor pure functions where possible
  • Keep methods focused and single-purpose

7. Code Quality Rules

  • Meaningful Naming: Use descriptive names for classes, methods, and variables that convey intent
  • Short Methods: Each method should do one thing; keep methods concise
  • No Duplication: Extract repeated logic into reusable abstractions (DRY principle)
  • Error Handling: Use exceptions rather than error codes; provide meaningful error messages

8. Anti-Patterns to Avoid

  • God Class: A class with too many responsibilities; split into focused classes
  • Spaghetti Code: Tangled, unstructured control flow; apply proper design
  • Magic Values: Hardcoded numbers or strings without constants; use named constants
  • Premature Optimization: Optimize only when measurement proves it necessary
  • Excessive Inheritance: Deep hierarchies create tight coupling; prefer composition
  • Over-Engineering: Creating unnecessary abstraction layers, interfaces, or packages that don't yet have clear use cases
  • Interface Duplication: Defining custom interfaces that duplicate standard library interfaces (e.g., defining your own Reader when io.Reader exists)
  • Premature Abstraction: Creating generic interfaces or base types before actual variation points are understood
  • Unused Code: "var _ Interface = (*Implementation)(nil)" style compile-time checks that serve no practical purpose

Source Transparency

This detail page is rendered from real SKILL.md content. Trust labels are metadata-based hints, not a safety guarantee.

Related Skills

Related by shared tags or category signals.

Coding

test-driven-development

No summary provided by upstream source.

Repository SourceNeeds Review
General

skill-creator

No summary provided by upstream source.

Repository SourceNeeds Review
General

eino-adk

No summary provided by upstream source.

Repository SourceNeeds Review
Coding

frontend-design

Create distinctive, production-grade frontend interfaces with high design quality. Use this skill when the user asks to build web components, pages, artifacts, posters, or applications (examples include websites, landing pages, dashboards, React components, HTML/CSS layouts, or when styling/beautifying any web UI). Generates creative, polished code and UI design that avoids generic AI aesthetics.

Repository SourceNeeds Review
161.1K94.2Kanthropics