Tabla de contenido

Buenas a todos,
En esta ocasión vamos a tratar sobre el análisis y el diseño orientados a objetos. Es fácil encontrar a muchos programadores que usan patrones de diseño como Singleton, Decorator, u Observer; pero no se han parado a analizarlos como tal.

Antes de entrar en materia con los patrones de diseño es importante entender conceptos básicos como son la abstracción, la encapsulación, el polimorfismo y la herencia, pero al mismo tiempo, es importante conocer los principios de diseño básicos para desarrollar de forma limpia y modular. La verdad, no hay nada peor que un código que sólo entienda su propio creador.

Hay por ahí algunos programadores que nunca han oído hablar sobre conceptos como OOPS (Object Oriented Programming System), o de los principios de diseño SOLID. Aunque también puede ser que no conozcan los beneficios que ofrecen estos principios o cómo usarlos a la hora de programar. O quizás sí que usan algunas técnicas pero no saben que tienen nombre. De todo puede pasar.

Podemos encontrar muy buenos ejemplos si echamos un vistazo a las librerias de Apache o Sun, donde nos enseñan como deben ser usados los principios de diseño en los programas Java, y en la programación en general.

La mejor manera de aprender cualquier principio o patrón es usando un ejemplo del mundo real para poder entenderlo de forma correcta y poder ver las consecuencias de no seguir esos principios. Podemos dejar claro ya que esto será una introducción. Probablemente se necesitaría un artículo para cada uno de estos puntos, pero vamos a dar un paseo en bici y así vemos un poco por encima los principios de diseño.

DRY (Don't Repeat Yourself o No te repitas)

Este será nuestro primer principio de diseño orientado a objetos, lo que podemos traducir como que no escribas código duplicado. Si os encontrais con que tenéis el mismo bloque de código en dos zonas distintas, deberíais considerar separalo en un método a parte. O quizás estais usando un valor fijo varias veces, lo mejor es usar una constante privada, o pública y estática, pero en ambos casos final.

Obviamente, el mayor beneficio de esto es el mantenimiento que ofrece. Aunque tampoco se debe abusar de esto. La duplicación no es por el código en sí, si no por la funcionalidad que ofrece. Vamos a ponernos en situación: estamos validando dos ids muy distintos usando un código muy parecido. La idea inicial es juntarlos en un mismo método y reutilizar. Pero no debe ser así. Vamos a pensar un poco...

Son dos validaciones distintas, y por tanto, el código ofrece dos funcionalidades distintas. ¿Qué pasaria si en un año el formato de uno de los ids cambia? Si nos hemos decantado por usado la misma función para validar ambos, al actualizar la función, uno de ellos dejará de funcionar. Esto se podría haber evitado.

El principio DRY se refiere a trozos de código que parece iguales y además hacen cosas iguales. Debe tenerse esto muy en cuenta.

Encapsular lo que puede cambiar

Lo único cierto que nos vamos a encontrar en el Software es que cambia. De modo que debemos encapsular lo que vemos, o sospechamos, que puede cambiar. Esto ofrece muchas ventajas, como por ejemplo pruebas y mantenimiento más sencillos. A la hora de programar en Java, lo mejor es hacer las variables y los métodos privados por defecto, y de ese modo aumentar el acceso paso a paso. Muchos patrones de diseño, como por ejemplo: Factory usa la encapsulación de este modo, dando flexibilidad de introducir nuevo código después sin impactar en el código existente.

El principio de diseño Cerrado-Abierto

Las clases, y los métodos o funciones, deberían estar abiertos para poder añadir nuevas funcionalidades, pero cerrados a las modificaciones. Esto nos permite no tener que modificar código que funciona y que ha sido probado. En una situación ideal, si debes añadir funcionalidad a una clase, tras terminar, únicamente deberías probar lo nuevo, y lo antiguo debería seguir funcionando de la misma forma que lo hacía antes. Esto se debe a que no deberías haber modificado nada de lo que había.

SRP (Single Responsibility Principle o Principio de responsabilidad única)

A la hora de modificar una clase, no debería existir más de una razón para llevarlo a cabo, o lo que es lo mismo, una clase únicamente debería dedicarse a una tarea bien concreta.

Si usas una clase para llevar a cabo más de una funcionalidad en Java, estarás introduciendo relaciones y/o uniones entre ambas. Lo peor de todo es que si aparece la necesidad de modificar una funcionalidad, lo más probable es que se rompa la otra. Esto requerirá reparar el problema, y luego una nueva tanda de pruebas para comprobar que todo está correcto. Podemos decir que se trata del doble de trabajo.

ISP (Interface Sgregation Principle o Principio de segregación de interfaces)

Este principio nos indica que una clase no debería implementar una interfaz si no la va usar. Esto ocurre sobre todo cuando una interfaz contiene más de una funcionalidad, y el cliente sólo necesita una de ellas.

El diseño de las interfaces puede ser complicado, dado que si creas una, ya no puedes modificarla sin romper todas las implementaciones realizadas. Pero uno de los beneficios de este principio de diseño es que, ya que Java obliga a implementar todos los métodos de la interfaz antes de usar una clase que la implemente, una sola funcionalidad significa menos métodos a implementar.

Programa usando interfaces y no implementaciones

Debes intentar siempre programa usando interfaces, esto te ayudará a obtener un código más flexible que podrá trabajar con cualquier nueva implementación de dichas interfaces.

De modo que sería razonable usar variables cuyo tipo sea una interfaz, y devolver en los métodos y usar en sus argumentos interfaces.

Principio de delegación

Intenta no hacer todo el trabajo por tu cuenta, delega cada tarea a la clase correspondiente. El ejemplo clásico usado del principio de delegación es la creación de los métodos equals() y hashCode(). A la hora de comparar dos objetos por igualdad siempre le preguntamos a la clase en sí que haga la comparación, en lugar de lo haga la clase padre (siempre que sea posible y correcto).

Todos estos principios de diseño pueden ayuda a conseguir un código mejor y más flexible, consiguiente una alta cohesión y baja relación entre las clases. La teoría es el primer paso, pero lo más importante es desarrollar estas habilidades para saberlas usar llegado el momento.

Debemos tener en cuenta que estos principios son realmente útiles para grandes proyectos y empresas que necesitan un mantenimiento muy largo. Nada es perfecto, de modo que tampoco intentes siempre resolver los problemas usando principio y patrones de diseño, simplemente son una buena práctica.

Espero comentarios.

Saludos,
Lázarus Surazal.

Entradas relacionadas

Perfil
prLázarus logo info

Carlos J. Peláez Rivas (Lázarus Surazal)

Graduado y Máster en Ingeniería Informática por la Universidad de Málaga (España). También cursé un Experto en tecnologías de Blockchain. Actualmente trabajando como Software engineer en Málaga.
Apasionado de los videojuegos, la música y la tecnología; siempre buscando cosas nuevas que aprender, hacer (y a veces romper).
Más sobre mi...
Contacto