Alta cohesión. Mantén tu código limpio y mantenible.
Java Backend CleanCode

Alta cohesión. Mantén tu código limpio y mantenible.

Silverio Martínez García
Silverio Martínez García

En el mundo del desarrollo de software, uno de los principios clave para crear aplicaciones robustas, mantenibles y escalables es la cohesión. La cohesión mide cuán estrechamente están relacionados los elementos dentro de una clase o módulo. La alta cohesión es un objetivo esencial en el diseño de clases y módulos en lenguajes de programación orientados a objetos como Java.

¿Qué es la Alta Cohesión?

En términos simples, alta cohesión significa que las responsabilidades de una clase o módulo están bien definidas y están estrechamente relacionadas entre sí. Una clase con alta cohesión tiene un propósito claro y está encargada de un conjunto limitado de tareas que están directamente relacionadas.

Por el contrario, una clase con baja cohesión tiene responsabilidades dispersas, lo que la hace difícil de entender, mantener y extender. Las clases altamente cohesionadas, en cambio, son más fáciles de comprender, probar y modificar, ya que sus métodos y atributos están fuertemente relacionados con un solo propósito.

Características de la Alta Cohesión:

  1. Responsabilidad única: La clase tiene una única razón para cambiar.
  2. Relación entre métodos y atributos: Los métodos y atributos de la clase están relacionados con la misma funcionalidad.
  3. Facilidad de mantenimiento: Las clases con alta cohesión son más fáciles de modificar sin afectar otras partes del sistema.
  4. Mayor reutilización: Las clases cohesionadas pueden ser reutilizadas más fácilmente en diferentes contextos, ya que su funcionalidad está claramente definida.

¿Por qué es importante la Alta Cohesión?

Mantener alta cohesión dentro de las clases tiene varios beneficios clave:

  • Mejora la legibilidad: Cuando una clase tiene un propósito claro y sus métodos son coherentes entre sí, es mucho más fácil de leer y entender.
  • Facilita las pruebas unitarias: Las clases con alta cohesión son más fáciles de probar porque se centran en una sola responsabilidad, lo que simplifica la creación de pruebas.
  • Simplifica el mantenimiento: Los cambios en una clase cohesionada son menos propensos a afectar otras partes del sistema.
  • Fomenta el diseño modular: Al tener clases que se encargan de responsabilidades bien definidas, se facilita la modularización y la separación de preocupaciones, lo que mejora la escalabilidad del sistema.

Ejemplo de Baja Cohesión

Supongamos que tienes una clase que maneja tanto la persistencia de datos como la lógica de negocio. Esta clase estaría haciendo varias cosas no relacionadas, lo que resulta en una baja cohesión.


public class UsuarioManager {

    // Método para gestionar la lógica de negocio de los usuarios
    public void registrarUsuario(String nombre, String email) {
        // Lógica para registrar al usuario
        System.out.println("Usuario registrado: " + nombre);
    }

    // Método para persistir los datos en la base de datos
    public void guardarUsuarioEnBaseDeDatos(String nombre, String email) {
        // Lógica para guardar en base de datos
        System.out.println("Usuario guardado en la base de datos: " + nombre);
    }

    // Método para enviar un correo al usuario
    public void enviarCorreoDeBienvenida(String email) {
        // Lógica para enviar correo
        System.out.println("Correo de bienvenida enviado a: " + email);
    }
}

En este ejemplo, la clase UsuarioManager está manejando múltiples responsabilidades: lógica de negocio, persistencia y comunicaciones (correo). Esto genera una baja cohesión, ya que no está centrada en un solo propósito. Si decidimos cambiar la forma en que enviamos correos, afectaría a la clase entera, lo que hace que el mantenimiento sea más complicado.

Ejemplo de Alta Cohesión

Ahora, veamos cómo mejorar la cohesión dividiendo las responsabilidades en clases más pequeñas y específicas.


public class UsuarioRegistrador {
    public void registrarUsuario(String nombre, String email) {
        // Lógica para registrar al usuario
        System.out.println("Usuario registrado: " + nombre);
    }
}

public class UsuarioRepositorio {
    public void guardarUsuarioEnBaseDeDatos(String nombre, String email) {
        // Lógica para guardar en base de datos
        System.out.println("Usuario guardado en la base de datos: " + nombre);
    }
}

public class CorreoDeBienvenida {
    public void enviarCorreo(String email) {
        // Lógica para enviar correo
        System.out.println("Correo de bienvenida enviado a: " + email);
    }
}

En este ejemplo, hemos dividido las responsabilidades en tres clases:

  1. UsuarioRegistrador: Se encarga exclusivamente de la lógica de negocio relacionada con el registro de usuarios.
  2. UsuarioRepositorio: Se ocupa de la persistencia de los usuarios en la base de datos.
  3. CorreoDeBienvenida: Se encarga de enviar los correos electrónicos de bienvenida.

Cada clase tiene una responsabilidad clara y bien definida, lo que hace que el código sea más fácil de mantener y extender. Si, por ejemplo, necesitamos cambiar la manera en que se envían los correos electrónicos, solo necesitaremos modificar la clase CorreoDeBienvenida sin tocar las otras clases.

Beneficios:

  • Facilidad de mantenimiento: Cada clase tiene un propósito claro, lo que reduce la posibilidad de introducir errores cuando realizamos cambios.
  • Mayor modularidad: Si queremos agregar nuevas funcionalidades, como un sistema de notificaciones, podemos hacerlo sin afectar el resto del código.
  • Mejor testabilidad: Al estar bien separadas las responsabilidades, se facilita la escritura de pruebas unitarias.

Cómo Lograr Alta Cohesión en Java

Para lograr alta cohesión en Java, hay varias estrategias que puedes seguir:

  1. Aplicar el principio de responsabilidad única (SRP): Cada clase debe tener una única razón para cambiar, lo que implica que debe centrarse en una sola tarea.
  2. Refactorizar cuando sea necesario: Si una clase empieza a asumir demasiadas responsabilidades, es momento de refactorizarla en clases más pequeñas y especializadas.
  3. Usar clases pequeñas y enfocadas: Asegúrate de que cada clase tenga un único propósito y que todos sus métodos estén relacionados con esa responsabilidad.
  4. Composición sobre herencia: A menudo, es más eficiente componer objetos con responsabilidades bien definidas que intentar heredarlas de una clase base con múltiples responsabilidades.

Conclusión

En resumen, la alta cohesión es un principio esencial para crear software bien estructurado y fácil de mantener. Al seguir este principio, podemos crear clases en Java que sean claras, modulares y fáciles de probar, lo que a su vez facilita la evolución del sistema sin generar complicaciones innecesarias.

Recuerda, una clase con alta cohesión tiene un propósito claro, se enfoca en tareas específicas y reduce la probabilidad de errores al hacer cambios.