Skip to content

Ejemplos simples del uso de patrones de diseño.

Notifications You must be signed in to change notification settings

EzeCuccorese/design-patterns

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

64 Commits
 
 
 
 
 
 

Repository files navigation

Patrones de Diseño en Java.

Introducción.

Christopher Alexander dice “Cada patrón describe un problema que ocurre una y otra vez en nuestro entorno, para luego proponer el núcleo de la solución, de tal forma que esa solución puede ser usada millones de veces sin siquiera hacerlo dos veces de la misma forma”.

En otra definición, “Los patrones de diseño son el esqueleto de las soluciones a problemas comunes en el desarrollo de software.”

La siguiente es una lista de patrones de diseño con descripciones breves y con aplicaciones en ejemplos muy sencillos de entender. En algunos casos se trata de recopilaciones de otros autores, a los que se podrá acceder para ampliar la información.

Patrones de Creación (Creational Patterns)

Relativos al proceso de creación de un objeto.

Patrones de Estructura (Structural Patterns)

Composición de clases u objetos.

Patrones de Comportamiento (Behavioral Patterns)

Forma en que las clases u objetos interaccionan y distribuyen funcionalidades.

Propósito: Definir una familia de algoritmos, encapsular cada uno, y que sean intercambiables. Strategy permite al algoritmo variar independientemente de los clientes que lo utilizan.

Aplicación: Usamos el patrón Strategy cuando queremos...

  • Definir una familia de comportamientos.
  • Definir variantes de un mismo algoritmo.
  • Poder cambiar el comportamiento en tiempo de ejecución, es decir, dinámicamente.
  • Reducir largas listas de condicionales.
  • Evitar código duplicado.
  • Ocultar código complicado, o que no queremos revelar, del usuario.

Ejemplos:

Propósito: Definir una dependencia de uno a muchos entre los objetos de manera que cuando un objeto cambia de estado, todos los que dependen de él son notificados y se actualizan automáticamente.

Los Observers se registran con el Subject a medida que se crean. Siempre que el Subject cambie, difundirá a todos los Observers registrados que ha cambiado, y cada Observer consulta al Subject que supervisa para obtener el cambio de estado que se haya generado.

En Java tenemos acceso a la clase Observer mediante java.util.Observer

Aplicación: Usamos el patrón Observer cuando...

  • Un cambio en un objeto requiere cambiar los demás, pero no sabemos cuántos objetos hay que cambiar.
  • Configurar de manera dinámica un componente de la Vista, envés de estáticamente durante el tiempo de compilación.
  • Un objeto debe ser capaz de notificar a otros objetos sin que estos objetos estén fuertemente acoplados.

Ejemplos:

El ejemplo de Subasta presenta una variante distinta de muchas fuentes en Internet de este patrón, usaremos una clase Evento para controlar los sucesos a los que los Observers responderan.

Propósito: Definir una interface para crear un objeto, dejando a las subclases decidir de que tipo de clase se realizará la instancia en tiempo de ejecución. Reducir el uso del operador new.

Crear objetos en una clase usando un método factory es más flexible que crear un objeto directamente. Es posible conectar la generación de familias de clases que tienen comportamientos en común. Elimina la necesidad de estar haciendo binding (casting) hacia clases específicas dentro del código, ya que éste sólo se entiende con las clases abstractas.

Aplicación: Usamos el patrón Factory...

  • Cuando una clase no puede anticipar que clase de objetos debe crear, esto se sabe durante el tiempo de ejecución.
  • Cuando un método regresa una de muchas posibles clases que comparten carecterísticas comunes a través de una superclase.
  • Para encapsular la creación de objetos.

Ejemplos:

Propósito: Proveer una interfaz para la creación de familias u objetos dependientes relacionados, sin especificar sus clases concretas.

Es una jerarquía que encapsula muchas familias posibles y la creación de un conjunto de productos. El objeto "fábrica" tiene la responsabilidad de proporcionar servicios de creación para toda una familia de productos. Los "clientes" nunca crean directamente los objetos de la familia, piden la fábrica que los cree por ellos.

Aplicación: Usamos el patrón Abstract Factory...

  • Cuando tenemos una o múltiples familias de productos.
  • Cuando tenemos muchos objetos que pueden ser cambiados o agregados durante el tiempo de ejecución.
  • Cuando queremos obtener un objeto compuesto de otros objetos, los cuales desconocemos a que clase pertenecen.
  • Para encapsular la creación de muchos objetos.

Ejemplos:

Propósito: Componer objetos en estructuras de árbol que representan jerarquías de un todo y sus partes. El Composite provee a los clientes un mismo trato para todos los objetos que forman la jerarquía.

Pensemos en nuestro sistema de archivos, este contiene directorios con archivos y a su vez estos archivos pueden ser otros directorios que contenga más archivos, y así sucesivamente. Lo anterior puede ser representado facilmente con el patrón Composite.

Aplicación: Usamos el patrón Composite...

  • Cuando queremos representar jerarquías de objetos compuestas por un todo y sus partes.
  • Se quiere que los clientes ignoren la diferencia entre la composición de objetos y su uso individual.

Ejemplos:

Propósito: Asegurar que una clase tenga una única instancia y proporcionar un punto de acceso global a la misma. El cliente llama a la función de acceso cuando se requiere una referencia a la instancia única.

Aplicación: Usamos el patrón Singleton...

  • La aplicación necesita una, y sólo una, instancia de una clase, además está instancia requiere ser accesible desde cualquier punto de la aplicación.
  • Tipicamente para:
    • Manejar conexiones con la base de datos.
    • La clase para hacer Login.

Ejemplos:

Propósito: Separar la construcción de un objeto complejo de su representación para que el mismo proceso de construcción puede crear diferentes representaciones.

Nos permite crear un objeto que está compuesto por muchos otros objetos. Sólo el "Builder" conoce a detalle las clases concretas de los objetos que serán creados, nadie más.

En este patrón intervienen un "Director" y un "Builder". El "Director" invoca los servicios del "Builder" el cual va creando las partes de un objeto complejo y al mismo tiempo guardo un estado intermedio de la construcción del objeto. Cuando el producto se ha construido por completo el cliente recupera el resultado.

A diferencia de otros patrones creacionales que construyen productos de una sola vez, el patrón "Builder" construye paso a paso los productos bajo el control del "Director".

Aplicación: Usamos el patrón Builder cuando queremos...

  • Construir un objeto compuesto de otros objetos.
  • Que la creación de las partes de un objeto sea independiente del objeto principal.
  • Ocultar la creación de las partes de un objeto del cliente, de esta manera no existe dependencia entre el cliente y las partes.

Ejemplos:

Propósito: Especificar varios tipos de objetos que pueden ser creados en un prototipo para crear nuevos objetos copiando ese prototipo. Reduce la necesidad de crear subclases.

Aplicación: Usamos el patrón Prototype...

  • Queremos crear nuevos objetos mediante la clonación o copia de otros.
  • Cuando tenemos muchas clases potenciales que queremos usar sólo si son requeridas durante el tiempo de ejecución.

Ejemplos:

Como cualquier adaptador en el mundo real este patrón se utiliza para ser una interfaz, un puente, entre dos objetos. En el mundo real existen adaptadores para fuentes de alimentación, tarjetas de memoria de una cámara, entre otros. En el desarrollo de software, es lo mismo.

Propósito: Convertir la interfaz (adaptee) de una clase en otra interfaz (target) que el cliente espera. Permitir a dos interfaces incompatibles trabajar en conjunto. Este patrón nos permite ver a nuevos y distintos elementos como si fueran igual a la interfaz conocida por nuestra aplicación.

Aplicación: Usamos el patrón Adapter...

  • Cuando el cliente espera usar la interfaz de destino (target).
  • Deseamos usar una clase existente pero la interfaz que ofrece no concuerda con la que necesitamos.

Ejemplos:

Extender la funcionalidad de los objetos se puede hacer de forma estática en nuestro código (tiempo de compilación) mediante el uso de la herencia, sin embargo, podría ser necesario extender la funcionalidad de un objeto de manera dinámica.

Propósito: Adjuntar responsabilidades adicionales a un objeto de forma dinámica. Los decoradores proporcionan una alternativa flexible para ampliar la funcionalidad.

Aplicación: Usamos el patrón Decorator...

  • Cuando necesitamos añadir o eliminar dinámicamente las responsabilidades a un objeto, sin afectar a otros objetos.
  • Cuando queremos tener las ventajas de la Herencia pero necesitemos añadir funcionalidad durante el tiempo de ejecución. Es más flexible que la Herencia,
  • Simplificar el código agregando funcionalidades usando muchas clases diferentes.
  • Evitar sobreescribir código viejo agregando, envés, código nuevo.

Ejemplos:

Propósito: Proporcionar una interfaz unificada para un conjunto de interfaces de un subsistema. Facade define una interfaz de alto nivel que hace que el subsistema sea más fácil de usar.

Detalles:

  • Este patrón protege los clientes de los componentes del subsistema, propiciando el menor uso de componentes para que el subsistema pueda ser utilizado.
  • Además, promueve un bajo acoplamiento entre subsistemas y clientes.
  • Este patrón no evita que los clientes usen las clases internas del subsistema, si es que es necesario.
  • Es importante mencionar que el objeto Facade debe ser extremadamente simple. No debe convertirse en un objeto "dios".

Aplicación: Usamos el patrón Facade...

  • Cuando queremos encapsular un subsistema complejo con una interface más simple.
  • Para crear una interface simplificada que ejecuta muchas acciones "detŕas del escenario".
  • Existen muchas dependencias entre clientes y la implementación de clases de una abstracción. Se introduce el facade para desacoplar el subsistema de los clientes y otros subsistemas.
  • Necesitamos desacoplar subsistemas entre sí, haciendo que se comuniquen únicamente mediante Facades.
  • Para definir un punto de entrada a cada nivel del subsistema.

Ejemplos:

Tengamos en cuenta el siguiente escenario: Es necesario instanciar objetos sólo cuando sean efectivamente solicitados (request) por el cliente.

Un Proxy o sustituto:

  1. Crea una instancia del objeto real la primera vez que el cliente realiza una solicitud del proxy.
  2. Recuerda la identidad de este objeto real.
  3. Finalmente, envía la solicitud del servicio al objeto real.

Propósito:

  • Proveer un sustituto o "placeholder" de otro objeto para controlar el acceso a este.
  • Usar un nivel extra de indirección para permitir el acceso distribuido, controlado e inteligente.
  • Agregar un "wrapper" para proteger el componente real de la complejidad innecesaria. Este wrapper permite agregar funcionalidad al objeto de interés sin cambiar el código del objeto.

Aunque a simple vista tenga similitudes al patrón Facade, ambos tiene matices diferentes.

  • Mientras que los objetos Proxy representan a un objeto, los objetos Facade representan a un subsistema de objetos.
  • Un objeto cliente Proxy no puede acceder a un objeto interno directamente, mientras que Facade no lo impide.
  • Un objeto Proxy provee control de acceso a un único objeto de interés. Sin embargo, un objeto Facade provee una interface de alto nivel a un subsistema de objetos.

Aplicación: Usamos el patrón Proxy...

  • Cuando haya necesidad de una referencia más versátil y sofisticada a un objeto, no un simple puntero.
  • Para adicionar seguridad al acceso de un objeto. El Proxy determinará si el cliente puede acceder al objeto de interés.
  • Para proporcionar una API simplificada para que el código del cliente no tenga que lidiar con la complejidad del código del objeto de interés.
  • Para proporcionar una interfaz de los web services o recursos REST.

Escenarios de uso:

  • Remote Proxy: Representa un objeto local que pertenece a un espacio de direcciones diferente.
  • Virtual Proxy: En lugar de un objeto complejo o pesado, utiliza una representación de esqueleto. Consideremos una imagen la cual es enorme en tamaño, podemos representarla mediante un objeto proxy virtual y cuando sea solicitado cargamos el objeto real.
  • Protection Proxy: Controla el acceso al objeto original. Este tipo es útil cuando se necesita manejar diferentes permisos de acceso.

Ejemplos:

El patrón Command encapsula comandos( llamados a métodos) en objetos, permitiéndonos realizar peticiones sin conocer exactamente la petición que se realiza o el objeto al cuál se le hace la petición. Este patrón nos provee las opciones para hacer listas de comandos, hacer/deshacer acciones y otras manipulaciones.

Este patrón desacopla al objeto que invoca la operación del objeto que sabe cómo llevar a cabo la misma. Un objeto llamado Invoker transfiere el comando a otro objeto llamado Receiver el cual ejecuta el código correcto para el comando recibido.

Propósito: Encapsular una petición en forma de objeto, permitiendo de ese modo que parametrizar clientes con diferentes peticiones, "colas" o registros de solicitudes, y apoyar las operaciones de deshacer.

Aplicación: Usamos el patrón Command...

  • Cuando queremos realizar peticiones en diferentes tiempos. Se puede hacer a través de la especificación de una "cola".
  • Para implementar la función de deshacer (undo), ya que se puede almacenar el estado de la ejecución del comando para revertir sus efectos.
  • Cuando necesitemos mantener un registro (log) de los cambios y acciones.

Usos típicos:

  • Mantener un historial de peticiones. (requests)
  • Implementar la funcionalidad de callbacks.
  • Implementar la funcionalidad de undo.

Ejemplos:

Propósito: Permite reutilizar objetos con el fin de evitar el costo de crearlos cada vez que la aplicación los requiera, manteniendo así un almacén de objetos creados previamente para ser utilizados.

Cuando un objeto es tomado del pool dejará de estar disponible hasta que se devuelva. Si no hay objetos disponibles, se crea uno nuevo y se agrega al conjunto. En algunos casos los recursos pueden ser limitados por lo que se especifica un número máximo de objetos. Si se alcanza este número y se solicita un nuevo elemento, puede lanzarse una excepción, o se puede bloquear el hilo de trabajo hasta que un objeto se libere de nuevo.

Aplicación: Usamos el patrón Object Pool cuando...

  • Se requiere trabajar con una gran cantidad de objetos, los cuales son computacionalmente caros de crear.
  • Se requiere trabjar con objetos por un tiempo muy corto y que luego de su uso son desechados.

Escenarios de uso:

  • Database connection: Si hay una necesidad de abrir demasiadas conexiones para una base de datos tomaría demasiado tiempo crear una nueva y el servidor de base de datos podría sobrecargarse.
  • Share resources: Diferentes clientes necesitarán el mismo recurso en diferentes momentos.

Referencia: https://www.javatpoint.com/object-pool-pattern

Propósito: Permite desacoplar una abstracción de su implementación, de manera que ambas puedan ser modificadas independientemente sin necesidad de alterar por ello la otra.

Aplicación: Usamos el patrón Bridge cuando...

  • Se desea evitar un enlace permanente entre la abstracción y su implementación. Esto puede ser debido a que la implementación debe ser seleccionada o cambiada en tiempo de ejecución.
  • Tanto las abstracciones como sus implementaciones deben ser extensibles por medio de subclases. En este caso, el patrón Bridge permite combinar abstracciones e implementaciones diferentes y extenderlas independientemente.
  • Cambios en la implementación de una abstracción no deben impactar en los clientes, es decir, su código no debe tener que ser recompilado.
  • Se desea compartir una implementación entre múltiples objetos (quizá usando contadores), y este hecho debe ser escondido a los clientes.

Usos típicos:

  • Desacoplar interfaz e implementación.
  • Mejorar la extensibilidad.
  • Esconder los detalles de la implementación a los clientes.

Ejemplos:

Referencia: https://www.geeksforgeeks.org/bridge-design-pattern/

Propósito: Permite eliminar o reducir la redundancia cuando tenemos gran cantidad de objetos que contienen información idéntica, además de lograr un equilibrio entre flexibilidad y rendimiento (uso de recursos).

Una característica importante es que los objetos Flyweight son inmutables, por lo que no podrán ser modificados una vez construidos. Puede utilizarse conjuntamente con el patrón Factory, de tal modo que en el momento en que se soliciten instancias del objeto, compruebe si ya existe un tipo Flyweight y devuelva esa referencia. En caso de que no exista la creará y la registrará.

Aplicación: Usamos el patrón Flyweight cuando...

  • Reducir el uso de memoria y/o evitar errores con la misma (java.lang.OutOfMemoryError).
  • Aunque la creación de objetos puede ser rápida, se podría reducir los tiempos de ejecución compartiendo objetos.

Ejemplos:

Referencia: https://www.geeksforgeeks.org/flyweight-design-pattern/

Propósito: Evita acoplar el emisor de una petición a su receptor dando a más de un objeto la posibilidad de responder a una petición.

Cada elemento deberá tener referencia al siguiente.

Aplicación: Usamos el patrón Chain of Responsability cuando...

  • Hay más de un objeto que puede manejar una petición, y el manejador no se conoce a priori, sino que debería determinarse automáticamente.
  • Se quiere enviar una petición a un objeto entre varios sin especificar explícitamente el receptor.
  • El conjunto de objetos que pueden tratar una petición debería ser especificado dinámicamente.

Ejemplos:

Propósito: Almacenar el estado de un objeto (o del sistema completo) en un momento dado, de manera que se pueda restaurar en ese punto de manera sencilla.

Se mantiene almacenado el estado del objeto para un instante de tiempo en una clase independiente de aquella a la que pertenece el objeto (pero sin romper la encapsulación), de forma que ese recuerdo permita que el objeto sea modificado y pueda volver a su estado anterior.

Aplicación: Usamos el patrón Memento cuando...

  • Se quiere poder restaurar el sistema desde estados pasados.
  • Se desea facilitar el hacer y deshacer de determinadas operaciones, para lo que habrá que guardar los estados anteriores de los objetos sobre los que se opere (o bien recordar los cambios de forma incremental).

Ejemplos:

Referencia: https://www.geeksforgeeks.org/memento-design-pattern/

Propósito: Define un objeto que encapsula cómo un conjunto de objetos interactúan, pudiendo alterar el comportamiento del programa en ejecución

La comunicación entre objetos es encapsulada con un objeto mediador. Los objetos no se comunican de forma directa entre ellos, en lugar de ello se comunican mediante el mediador. Esto reduce las dependencias entre los objetos en comunicación, reduciendo entonces la Dependencia de código.

Aplicación: Usamos el patrón Mediator cuando ...

  • Se desea reducir la dependencia entre clases evitando que los objetos se relacionen entre ellos de forma explícita, y permitiendo variar cualquier interacción independientemente.

Ejemplos:

Referencia: https://www.geeksforgeeks.org/mediator-design-pattern/

Propósito: Define el esqueleto de programa de un algoritmo en un método, llamado método de plantilla, el cual difiere algunos pasos a las subclases.

Permite redefinir ciertos pasos seguros de un algoritmo sin cambiar la estructura del algoritmo.

Aplicación: Usamos el patrón Template Method cuando ...

  • Se desea dejar que las subclases que se implementan (a través del método primordial) tengan un comportamiento que puede variar.
  • Evitar duplicación en el código: la estructura general de flujo de trabajo, está implementada una vez en el algoritmo de clase abstracta, y variaciones necesarias son implementadas en cada de las subclases.
  • Controlar en qué punto(s) la subclassing está permitida. En oposición a una sencilla sobrecarga polimórfica, donde el método de base sería enteramente reescrito, permitiendo un cambio radical en el flujo. Sólo los detalles específicos del flujo se pueden cambiar.

Ejemplos:

Referencia: https://www.geeksforgeeks.org/template-method-design-pattern/

Propósito: Define una interfaz que declara los métodos necesarios para acceder secuencialmente a un grupo de objetos de una colección.

Algunos de los métodos que podemos definir en la interfaz Iterador son:

Primero(), Siguiente(), HayMas() y ElementoActual().

Aplicación: Usamos el patrón Iterator cuando ...

  • Se desea acceder a los elementos de un contenedor de objetos (por ejemplo, una lista) sin exponer su representación interna.

Ejemplos:

Referencia: https://www.geeksforgeeks.org/iterator-pattern/

Propósito: Separar el algoritmo de la estructura de un objeto.

La idea básica es que se tiene un conjunto de clases elemento que conforman la estructura de un objeto. Cada una de estas clases elemento tiene un método aceptar (accept()) que recibe al objeto visitante (visitor) como argumento. El visitante es una interfaz que tiene un método visit diferente para cada clase elemento; por tanto habrá implementaciones de la interfaz visitor de la forma: visitorClase1, visitorClase2... visitorClaseN. El método accept de una clase elemento llama al método visit de su clase.

Si hay demasiadas implementaciones de la interface visitor, se hace dificil extender.

Aplicación: Usamos el patrón Visitor cuando ...

  • Se desea mover la lógica operacional desde los objetos a otra clase.
  • Se desea definir una operación sobre objetos de una jerarquía de clases sin modificar las clases sobre las que opera.
  • Se desea representar una operación que se realiza sobre los elementos que conforman la estructura de un objeto
  • Teniendo un buen número de instancias de un pequeño número de clases, se desea realizar alguna operación que involucra a todas o a la mayoría de ellas.

Ejemplos:

Propósito: Permite a un objeto alterar su comportamiento dependiendo de su estado interno.

Aplicación: Usamos el patrón State cuando ...

  • Un determinado objeto tiene diferentes estados y también distintas responsabilidades según el estado en que se encuentre en determinado instante.
  • Se desea simplificar casos en los que se tiene un complicado y extenso código de decisión que depende del estado del objeto.

Ejemplos:

Referencia: https://www.geeksforgeeks.org/state-design-pattern/

Propósito: Define una representación para su gramática junto con un intérprete del lenguaje.

Aplicación: Usamos el patrón Interpreter cuando ...

  • Se desea representar expresiones regulares que representen cadenas a buscar dentro de otras cadenas..
  • Para definir un lenguaje que permita representar las distintas instancias de una familia de problemas.

Ejemplos:

Referencia: https://www.baeldung.com/java-interpreter-pattern

About

Ejemplos simples del uso de patrones de diseño.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • Java 100.0%