Francisco Javier Palacios Pérez Fco. Javier Palacios Pérez
Software Developer
Why containers exist: from physical servers to Docker

Why containers exist: from physical servers to Docker

Why containers exist: from physical servers to Docker

Why containers exist: from physical servers to Docker

“It works on my machine.”

Four words that have ruined more Friday afternoons than anything else in the history of software development. If you’re a developer, you’ve said them. If you’re DevOps, you’ve heard them — probably followed by that uncomfortable silence where everyone knows the release is going to slip.

Docker didn’t appear out of nowhere. It appeared because that problem is structurally hard with previous approaches, and for decades there was no elegant solution. Before touching a single docker command, it’s worth understanding why.

The software deployment problem

Imagine you’re developing a web application on your laptop. You have Python 3.12, PostgreSQL 16, Redis 7, and a system library you installed six months ago that you can barely remember how to set up. Your application works perfectly.

Deployment time comes. The production server has Python 3.9 (because the infrastructure team “doesn’t touch what works”), PostgreSQL 14, and that system library you need isn’t installed. Or it’s a different version. Or it’s installed but in a different path.

What do you do? The same thing everyone has always done: document the installation steps, hope that production is close enough to your environment, and spend the next week debugging issues that only appear “in production.”

This is the environment dependency problem. It’s not a bug in your code — your code is fine. The problem is that software needs a specific context to run, and reproducing that context across different machines is surprisingly hard.

The physical server era

In the 90s and early 2000s, the model was simple: one application, one physical server. Three applications meant three servers.

[App A + dependencies + OS] → Server 1
[App B + dependencies + OS] → Server 2
[App C + dependencies + OS] → Server 3

The obvious problem is cost — physical servers are expensive and most of the time they’re underutilized. But there’s a less obvious problem: dependency conflicts.

If you try to run two applications on the same server, they might need different versions of the same library. Or one app’s network configuration interferes with the other’s. Or one app’s processes consume the other’s resources at the worst possible moment.

The solution was to give each application its own server. Safe, but expensive and wasteful.

The virtual machine revolution

By the mid-2000s, virtual machines (VMs) changed the game. The idea is elegant: on top of the physical hardware you install a hypervisor that can run multiple complete operating systems simultaneously, isolated from each other.

┌────────────────────────────────────────────────┐
│  [App A + Guest OS]  [App B + Guest OS]        │
│                  Hypervisor                    │
│              Host Operating System             │
│                   Hardware                     │
└────────────────────────────────────────────────┘

Now you can run ten applications on one physical server, each in its own VM with its own isolated operating system. If App A needs Ubuntu 18.04 with Python 3.6, and App B needs CentOS 7 with Python 3.9, no problem — each VM has its own world.

The isolation is complete. Hardware costs are distributed better. It was a huge leap.

But there’s a price: each VM carries a full operating system. A guest OS can take 1–2 GB of disk space and hundreds of MB of RAM, even before your application starts. If you have 20 applications, you’re paying the cost of 20 operating systems. Booting a VM takes minutes.

For many use cases, this is acceptable. But for environments where you need to scale fast, spin up dozens of instances in seconds, or have reproducible development environments that don’t eat your laptop’s RAM, VMs are too heavy.

Containers: isolation without the weight

This is where Docker (and containers in general) change the equation.

The core idea: instead of virtualizing hardware to run separate operating systems, containers virtualize at the operating system level. All containers share the same host OS kernel, but each has its own completely isolated filesystem, processes, network, and users.

┌─────────────────────────────────────────────────┐
│  [App A + deps]  [App B + deps]  [App C + deps] │
│              Container Runtime                  │
│              Host Operating System              │
│                    Hardware                     │
└─────────────────────────────────────────────────┘

The result: a container starts in milliseconds (not minutes), takes up megabytes (not gigabytes), and still has the same process, network, and filesystem isolation you need for “it works on my machine” to stop being a curse.

Why this actually solves the problem

The magic isn’t just isolation. It’s that Docker lets you package your application together with its entire environment into a single portable unit called an image.

That image includes: the base OS (only what’s needed, not a full OS), your language runtime at the exact version you need, all system libraries and dependencies, configuration files, and your code.

When anyone runs that image — on their laptop, on the staging server, in production on AWS — they get exactly the same environment. The same Python. The same libraries. The same configuration.

“It works on my machine” stops being a problem because everyone uses the same machine: the container.

Why Docker in 2026

Containers aren’t new — Linux has had the underlying primitives (cgroups, namespaces) for years. But Docker, which appeared in 2013, democratized their use with an understandable interface and a tooling ecosystem that made going from “command in my terminal” to “running in production” reasonably straightforward for the first time.

Today in 2026, containers are the foundation on which practically the entire modern infrastructure ecosystem is built:

  • Kubernetes orchestrates containers at scale
  • Modern CI/CD (GitHub Actions, GitLab CI) runs every pipeline in containers
  • Cloud providers have dedicated services for running containers (ECS, Cloud Run, Azure Container Instances)
  • Local development with reproducible environments is done with Docker Compose
  • Microservices are deployed as independent containers

If you want to work as a backend developer, DevOps engineer, SRE, or in any role that touches infrastructure, understanding containers isn’t optional. It’s the common vocabulary of the industry.

Key concepts from this lesson

  • The problem: environment dependencies make software work in one place and fail in another
  • Physical servers: one app per server, expensive and inefficient
  • Virtual machines: better hardware utilization, but heavy (full OS per VM)
  • Containers: OS-level isolation, shared kernel, millisecond startup
  • Docker image: the portable package that includes app + complete environment
  • The promise: same container = same behavior in any environment

In the next lesson, lesson 2, we move from theory to practice: we’ll install Docker on your machine (Linux, macOS, or Windows with WSL) and run our first container.

Never stop coding!