Ir al contenido principal

Open/Closed Principle (OCP) – SOLID explicado con ejemplos

Continuando con el repaso de los principios de S.O.L.I.D. que inició en el hilo anterior - si no lo viste hacé click acá Principio de Responsabilidad Única (SRP) - vamos a ver el segundo en orden de aparición: Principio de Abierto/Cerrado (OCP por sus siglas en inglés).

Definición Formal

El principio OCP (Open/Closed Principle) establece que el código de una clase o un módulo debe estar abierto para la extensión, pero cerrado para la modificación. Esto significa que no se deben realizar cambios en el código existente cuando se requiere alterar alguna funcionalidad. En lugar de modificar el código existente, se debe crear una nueva implementación que extienda la funcionalidad. La única excepción a esto son los arreglos de bugs, donde está permitido modificar el código existente.

Si se desea introducir una nueva funcionalidad, como la ordenación en un método existente, en lugar de modificar el código, se crearía una nueva implementación que implementara la misma interfaz.

El principio OCP fomenta la creación de código más robusto y flexible, minimizando la introducción de errores en el sistema. Siguiendo también el principio de segregación de interfaces, se pueden crear interfaces más pequeñas, lo que facilitaría la implementación de nuevos servicios sin tener que reescribir todo el código. La combinación de OCP e IS (Interface Segregation) permite que los desarrolladores mantengan un código limpio y escalable.

Las ventajas de no modificar el código existente, permite que los casos de prueba que ya están escritos para la implementación sigan siendo relevantes y no necesiten revisión. Esto ahorra tiempo y esfuerzo en producción, y permite a los clientes seguir usando la funcionalidad antigua si así lo desean.

Bertrand Meyer fue quien formuló originalmente el Principio Abierto/Cerrado (OCP) en su libro "Object-Oriented Software Construction" (1988).

Definición de Meyer:

Una entidad de software (clase, módulo, función, etc.) debe estar abierta para extensión, pero cerrada para modificación.
Luego, este principio fue adoptado por Robert C. Martin (tío Bob) y pasó a formar parte de los Principios SOLID, que son muy relevantes en la programación orientada a objetos.

En resumen, Meyer lo introdujo y Martin lo popularizó dentro del marco de SOLID

Ejemplo: Cálculo de áreas

Supongamos que queremos calcular el área de diferentes figuras geométricas sin modificar el código existente cuando agregamos una nueva figura.

Código sin OCP (violando el principio)

En este caso, cada vez que agregamos una nueva figura, debemos modificar la clase AreaCalculator, lo que viola el OCP porque el código no está cerrado a modificaciones:
public class Circulo
{
    public double Radio { get; }

    public Circulo(double radio)
    {
        Radio = radio;
    }
}

public class Rectangulo
{
    public double Ancho { get; }
    public double Alto { get; }

    public Rectangulo(double ancho, double alto)
    {
        Ancho = ancho;
        Alto = alto;
    }
}

public class CalculadoraDeArea
{
    public double CalcularArea(object figura)
    {
        if (figura is Circulo c)
        {
            return 3.14 * c.Radio * c.Radio;
        }
        else if (figura is Rectangulo r)
        {
            return r.Ancho * r.Alto;
        }
        else
        {
            throw new ArgumentException("Figura no soportada");
        }
    }
}
Si queremos agregar una nueva figura (Triángulo), tendríamos que modificar la CalculadoraDeArea, violando el OCP.

Ejemplo Correcto (Cumple OCP)

Acá usamos herencia y polimorfismo para que la CalculadoraDeArea no necesite modificaciones al agregar nuevas figuras.
// Definimos una interfaz para garantizar que todas las figuras implementen el método CalcularArea()
public interface IFigura
{
    double CalcularArea();
}

// Creamos la clase Circulo que implementa la interfaz IFigura
public class Circulo : IFigura
{
    public double Radio { get; }

    public Circulo(double radio)
    {
        Radio = radio;
    }

    public double CalcularArea()
    {
        return Math.PI * Radio * Radio;
    }
}

// Creamos la clase Rectangulo que implementa la interfaz IFigura
public class Rectangulo : IFigura
{
    public double Ancho { get; }
    public double Alto { get; }

    public Rectangulo(double ancho, double alto)
    {
        Ancho = ancho;
        Alto = alto;
    }

    public double CalcularArea()
    {
        return Ancho * Alto; 
    }
}

// Nueva funcionalidad: Agregamos la clase Triángulo sin modificar el código de las otras figuras
public class Triangulo : IFigura
{
    public double Base { get; }
    public double Altura { get; }

    public Triangulo(double baseT, double altura)
    {
        Base = baseT;
        Altura = altura;
    }

    public double CalcularArea()
    {
        return (Base * Altura) / 2; 
    }
}

// Uso del código en el método Main
class Program
{
    static void Main()
    {
        // Creamos un círculo con radio 5
        Circulo circulo = new Circulo(5);
        Console.WriteLine("El área del círculo es: " + circulo.CalcularArea());

        // Creamos un rectángulo de 4x6
        Rectangulo rectangulo = new Rectangulo(4, 6);
        Console.WriteLine("El área del rectángulo es: " + rectangulo.CalcularArea());

        // Creamos un triángulo de base 3 y altura 4
        Triangulo triangulo = new Triangulo(3, 4);
        Console.WriteLine("El área del triángulo es: " + triangulo.CalcularArea());
    }
}
Si queremos agregar un Triángulo, solo creamos una nueva clase Triangulo sin modificar CalculadoraDeArea. El código es más mantenible y flexible, porque es modular. Este ejemplo cumple con el principio de OC, ya que CalculadoraDeArea está cerrada para modificaciones y abierta para extensiones.

Desventajas del OCP

Aaunque el Principio Abierto/Cerrado (OCP) ofrece muchas ventajas, también tiene algunas desventajas que es importante considerar, no todo es perfecto en la vida.

Aumento en la complejidad del código

  • Para cumplir con OCP, a menudo se necesita más abstracción (interfaces, clases base, polimorfismo), lo que puede hacer que el código sea más difícil de entender y mantener.
  • En proyectos pequeños, esta estructura puede ser innecesaria y sobrecomplicar la solución.

Mayor cantidad de clases e interfaces

  • Aplicar OCP correctamente suele implicar la creación de múltiples interfaces y clases nuevas cada vez que se requiere un cambio.
  • Esto puede llevar a una explosión de clases y archivos, dificultando la navegación del código.

Posible sobrecarga de memoria o rendimiento

  • En algunos casos, el uso excesivo de abstracciones puede impactar el rendimiento debido a la sobrecarga de instancias de objetos y llamadas a métodos virtuales o inyección de dependencias.

Mayor esfuerzo inicial en el diseño

  • Implementar OCP correctamente requiere pensar en la extensibilidad desde el principio, lo que puede aumentar el tiempo de desarrollo inicial.
  • Si no se tiene un diseño adecuado, se pueden crear estructuras innecesariamente complejas que nunca se van a reutilizar.

Dificultad en cambios de gran porte

  • Si bien OCP permite agregar nuevas funcionalidades sin mdificar el código existente, si la estructura base no está bien diseñada, extenderla puede volverse complicado y requerir una refactorización importante.

Cuándo aplicar OCP y cuándo no

  • Usalo cuando esperas que tu sistema crezca y necesite nuevas funcionalidades sin afectar el código existente.
  • Evítalo en proyectos pequeños donde la flexibilidad futura no es una preocupación inmediata.
En conclusión, OCP es un principio potente que mejora la mantenibilidad y escalabilidad, pero debe usarse con criterio para evitar sobrecomplicaciones innecesarias cuando no lo amerita. Tampoco hay que obsesionarse.

Otros artículos

Principio de Responsabilidad Única (SRP) – SOLID explicado con ejemplos

Introducción a S.O.L.I.D En esta entrada, intentaremos abordar un nuevo ciclo de conceptos que tienen que ver sobre los principios SOLID , que son fundamentales en la programación orientada a objetos o POO . Estos principios fueron formulados, en principio, por Robert C. Martin , también conocido como " Uncle Bob ", con el objetivo de mejorar la mantenibilidad y escalabilidad del código de software. SOLID es un acrónimo que representa cinco principios de diseño: Single Responsibility Principle (SRP) – Principio de Responsabilidad Única Open/Closed Principle (OCP) – Principio de Abierto/Cerrado Liskov Substitution Principle (LSP ) – Principio de Sustitución de Liskov Interface Segregation Principle (ISP) – Principio de Segregación de Interfaces Dependency Inversion Principle (DIP) – Principio de Inversión de Dependencias Estos principios nos ayudan a crear software más ...

Roadmap para Desarrolladores Backend en .NET en 2025

El mundo del desarrollo backend está en constante evolución, y mantenerse actualizado con las mejores prácticas y tecnologías es clave para seguir siendo competitivo en el mercado. Si estás buscando una guía clara y estructurada para mejorar tus habilidades en . NET backend , el sitio roadmap.sh ofrece un excelente punto de partida. Para esta entrada vamos a explorar lo siguiente: ¿Qué es roadmap.sh y por qué es relevante? El roadmap backend para .NET en 2025 Tecnologías y habilidades esenciales para backend a considerar ¿Qué es roadmap.sh y quién lo creó? roadmap.sh es una plataforma ampliamente reconocida dentro de la comunidad de desarrolladores. Fue creada por Kamran Ahmed, un Google Developer Expert y contribuidor en múltiples proyectos de código abierto. Desde su lanzamiento, la plataforma ha crecido exponencialmente, acumulando más de 300,000 estrellas en GitHub y una comunidad activa de...

Codewars

Como algunos ya saben, soy un gran entusiasta de las plataformas en línea dedicadas a la educación y al entrenamiento de habilidades, lo que podríamos llamar "gimnasios mentales". Hoy quiero presentarles, o tal vez recordarles, una de mis favoritas: Codewars, conocida también como " Guerras de Código " en español. Si aún no la conocían, los invito a explorarla. Y si ya la conocían, este es un buen momento para redescubrirla y sacarle más provecho. ¿Qué es Codewars? Codewars es una plataforma educativa en línea diseñada como un juego para entrenar habilitades de programación. fue fundada en noviembre de 2012 por Nathan Doctor y Jake Hoffner . La idea surgió durante una competencia de Startup Weekend ese mismo año, donde desarrollaron un prototipo que obtuvo el primer lugar. En la actualidad, Codewars es propiedad de Qualified , una empresa tecnológica que ofrece una plataforma para evaluar y entrenar habilidades en ingeniería de software. Con esta herramienta pod...

Principio de Sustitución de Liskov (LSP) – SOLID explicado con ejemplos

¿Qué es el Principio de Sustitución de Liskov? Imaginá que tenés un control remoto universal diseñado para funcionar con cualquier televisor. Si un nuevo modelo de TV no responde a los mismos comandos, el control deja de ser útil. El Principio de Sustitución de Liskov es como una garantía de que cualquier 'televisor' (o clase derivada) va a funcionar correctamente con el 'control remoto' (o clase base). El Principio de Sustitución de Liskov ( LSP ) es el tercer principio de SOLID , representado por la letra L y establece que: Los objetos de una clase derivada deben poder sustituir a los objetos de su clase base sin afectar el comportamiento correcto del programa. En otras palabras, si una clase hija hereda de una clase padre, cualquier instancia de la clase hija debería poder usarse en lugar de una instancia de la clase padre sin alterar la funcionalidad esperada. ¿Quién es Bárbara Liskov? Bárbara Liskov es una destacada científi...

Principio de Segregación de Interfaces (ISP) – SOLID explicado con ejemplos

El Principio de Segregación de Interfaces es otro de los principios SOLID y establece que: Una clase no debería verse obligada a depender de métodos que no utiliza .  En otras palabras, en lugar de crear una interfaz grande con muchos métodos, es mejor dividirla (segregar) en interfaces más pequeñas y específicas. Pero.. ¿Por qué? Obliga a implementar métodos innecesarios Si una clase solo necesita una "parte" de la funcionalidad de una interfaz, pero esta interfaz es muy grande, se va a ver obligada a implementar métodos que no usa. Esto es casi que inevitable si no buscamos la manera de separar mejor las responsabilidades. Imaginemos una interfaz IVehiculo que tiene los siguientes métodos: public interface IVehiculo { void Conducir () ; void Volar () ; void Navegar () ; } Si una clase Auto implementa esta interfaz, se ve obligado a definir métodos como Volar() o Navegar() , aunque un auto no vuela ni navega. public c...
Buy Me A Coffee