DEV Community

Utkuhan Akar
Utkuhan Akar

Posted on

Why I Stopped Using Singletons (And How It Saved Our Architecture and My Sanity)

Let’s be completely honest with ourselves. We’ve all been there. You are building a system, you need an object to be accessible from everywhere—a game manager, an API client, or a configuration service—and a voice whispers in your ear: "Just make it a Singleton. It’s clean, it’s fast, and Instance.DoSomething() is right there."

It feels like a superpower. Until the project scales.

Last sprint, we were adding automated tests to our core modules, and guess what happened? Our test suite started failing randomly. Not because the logic was wrong, but because our "convenient" Singletons were leaking state between tests like a broken pipeline.

That was the moment I said: "Okay, the honeymoon is over. We need a real architectural strategy here."

Here is how we moved away from global state tight-coupling, and why this shift is a complete win-win for both codebase health and team velocity.

The Inciting Incident: The High Cost of "Global" Convenience
From a business perspective, code quality is directly tied to maintenance costs. If changing a feature in Module A breaks Module B because they secretly share a global instance, your development velocity plummets.

When you use a classic Singleton, you are essentially hiding dependencies. Look at this approach:

// The "Convenient" Trap
public class PlayerController {
    public void TakeDamage(int amount) {
        // Hiding the dependency inside the method. 
        // As you know, code review is going to be lively when the Tech Lead sees this.
        GameManager.Instance.ReduceScore(amount); 
    }
}
Enter fullscreen mode Exit fullscreen mode

Why this status quo wasn't enough:

  1. Zero Testability: You can't mock GameManager.Instance easily. If you want to test PlayerController in isolation, you are forced to bring the entire game state with you.

  2. Hidden Coupling: Components look independent on the outside, but under the hood, they are tightly coupled.

  3. Concurrency Nightmares: (And yes, we learned this the hard way during multi-threading experiments) when two threads try to orchestrate state updates on a single instance simultaneously... well, let's just say my coffee consumption doubled that day.

The Pivot: Embracing Explicit Dependency Injection (DI)
We decided to put on our "Lead Architect" hats and refactor the core communication layer. The goal was simple: No more hiding. If a class needs a service to function, it must demand it explicitly through its constructor.

Here is the modular framework we moved towards:

// Step 1: Define the boundary via an Interface
public interface IGameManager {
    void ReduceScore(int amount);
}

// Step 2: Inject the dependency explicitly
public class PlayerController {
    private readonly IGameManager _gameManager;

    // The intent is now clear, self-documenting, and honest.
    public PlayerController(IGameManager gameManager) {
        _gameManager = gameManager ?? throw new ArgumentNullException(nameof(gameManager));
    }

    public void TakeDamage(int amount) {
        _gameManager.ReduceScore(amount);
    }
}
Enter fullscreen mode Exit fullscreen mode

Pro Tip (The "Oops" Moment): When we first started refactoring, we tried to build our own custom DI container from scratch because, hey, how hard could it be? (Classic engineer over-optimization reflex, right?). It ended up being a memory-leaking monster. We quickly pivoted, deleted that boilerplate code, and integrated a lightweight container/service provider. Lesson learned: utilize battle-tested architectural patterns instead of reinventing the wheel.

The Decision Matrix: Trade-offs Matter
When you are pitching an architectural shift to stakeholders or tech leads, you don't just say "it feels cleaner." You present the trade-offs. Here is how the matrix looked for us:

Business Impact & Takeaways
At the end of the day, high-quality engineering must translate into a high-quality product. By introducing strict separation of concerns and explicit dependency orchestration, we achieved three things:

Onboarding Velocity: New developers can look at a class constructor and immediately understand its bandwidth and requirements without digging through thousands of lines of code.

Flawless Test Pipelines: Our CI/CD pipeline runs unit tests in parallel now. No shared state, no flaky test results. Just green checkmarks.

Production Peace of Mind: We can now swap implementations (e.g., changing a local save system to a cloud-based one) by modifying a single line in our composition root, rather than refactoring 50 different files.

If your codebase is currently slightly on fire because of global states, take a deep breath. Stop writing features for a second, define your system boundaries, and start decoupling. Your future self (and your tech lead) will thank you.

What about you? What is your go-to strategy for handling global state when a project starts outgrowing its initial design? Let’s discuss in the comments below!

Top comments (0)