Ir al contenido principal

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 modular, flexible y fácil de mantener. En este artículo, nos vamos a enfocar en el primero de ellos: el Principio de Responsabilidad Única (SRP).

Así que vamos allá.


¿Qué es el Principio de Responsabilidad Única (SRP)?

El Principio de Responsabilidad Única establece que una clase debe proporcionar solo una funcionalidad, no más de una.

Por ejemplo, si una clase implementa validaciones, recuperación de datos desde la configuración y acceso a la base de datos, está manejando múltiples responsabilidades, lo cual es una mala práctica. En su lugar, cada funcionalidad debe implementarse en una clase separada: una para acceso a datos, otra para validaciones, otra para manejo de caché, otra para manipulación de cookies, y otra para manejar solicitudes de usuario.

El problema de incluir múltiples funcionalidades en una sola clase es que el código se vuelve difícil de mantener, depurar e identificar errores. Además, si se necesita extender o modificar una funcionalidad existente, habría que hacer cambios en la misma clase, lo que podría afectar otras funcionalidades. Esto obligaría a reescribir casos de prueba, ya que los existentes podrían no ser relevantes con el código actualizado. También complica el trabajo en equipo, ya que sería difícil dividir tareas en equipos independientes.

Para evitar estos problemas, el SRP recomienda que una clase tenga solo un motivo para cambiar, es decir, que implemente una única funcionalidad.

Como beneficio, las clases serán independientes, fáciles de entender, modificar, diseñar, escribir, mantener y probar.

Un ejemplo típico de violación del SRP es una clase que implementa tanto la lectura de configuraciones como validaciones de modelos. En este caso, es recomendable dividir el código en dos clases separadas:
  • Una clase para la validación.
  • Otra clase para la lectura de configuración.
Sin embargo, esto no significa que una clase no pueda llamar a métodos de otra clase. Por ejemplo, una clase puede llamar un método de otra cuando sea necesario, pero sin acoplarse completamente a su código.

Veamos un mal ejemplo en C# donde se viola este principio:
public class ReportManager
{
    public void GenerateReport()
    {
        // Genera el informe
        Console.WriteLine("Generando el informe...");
    }

    public void SaveToFile(string content)
    {
        // Guarda el informe en un archivo
        File.WriteAllText("reporte.txt", content);
    }

    public void SendEmail(string email)
    {
        // Envía el informe por correo
        Console.WriteLine($"Enviando el informe a {email}");
    }
}
Acá, la clase ReportManager tiene tres responsabilidades:
  1. Generar un informe.
  2. Guardar el informe en un archivo.
  3. Enviar el informe por correo.
Esto hace que la clase sea difícil de mantener, ya que un cambio en la lógica de almacenamiento o en la lógica de envío podría afectar la generación del informe.

Aplicando el Principio de Responsabilidad Única

Para corregir el problema del ejemplo anterior, podemos separar las responsabilidades en clases independientes:
public class ReportGenerator
{
    public string GenerateReport()
    {
        return "Este es un informe generado.";
    }
}

public class ReportSaver
{
    public void SaveToFile(string content)
    {
        File.WriteAllText("reporte.txt", content);
    }
}

public class EmailService
{
    public void SendEmail(string email, string content)
    {
        Console.WriteLine($"Enviando el informe a {email}");
    }
}
Ahora cada clase tiene una única responsabilidad, haciendo que el código sea más modular y fácil de modificar sin afectar otras partes del sistema.

Finalmente, podemos coordinar estas clases en un servicio central:
public class ReportManager
{
    private readonly ReportGenerator _generator;
    private readonly ReportSaver _saver;
    private readonly EmailService _emailService;

    public ReportManager()
    {
        _generator = new ReportGenerator();
        _saver = new ReportSaver();
        _emailService = new EmailService();
    }

    public void ProcessReport(string email)
    {
        var report = _generator.GenerateReport();
        _saver.SaveToFile(report);
        _emailService.SendEmail(email, report);
    }
}
Ahora, si necesitamos cambiar la forma en que se genera el informe, almacenamos los archivos o enviamos correos electrónicos, podemos hacerlo sin afectar el resto del código.

El Principio de Responsabilidad Única nos ayuda a estructurar mejor nuestro código, facilitando su mantenimiento y evitando problemas derivados de la alta interdependencia entre funciones. En el próximo artículo, vamos a ver qué nos plantea el Principio de Abierto/Cerrado (OCP), que es el segundo en aparición en S.O.L.I.D. Hasta la próxima.

Otros artículos

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 q...

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