From 9309b5fda48c9a487c29bef508df0309f2835b9a Mon Sep 17 00:00:00 2001 From: peterm85 Date: Thu, 21 Nov 2019 11:26:38 +0100 Subject: [PATCH] Object pool pattern --- README.md | 577 +++++++++++---------- src/objectpool/pattern/Client.java | 29 ++ src/objectpool/pattern/HeavyObject.java | 26 + src/objectpool/pattern/ObjectPool.java | 32 ++ src/objectpool/pattern/ObjectPoolTest.java | 49 ++ 5 files changed, 443 insertions(+), 270 deletions(-) create mode 100644 src/objectpool/pattern/Client.java create mode 100644 src/objectpool/pattern/HeavyObject.java create mode 100644 src/objectpool/pattern/ObjectPool.java create mode 100644 src/objectpool/pattern/ObjectPoolTest.java diff --git a/README.md b/README.md index 7f93c08..6907976 100644 --- a/README.md +++ b/README.md @@ -1,270 +1,307 @@ -# 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 descripcipnes breves y con aplicaciones en ejemplos muy sencillos de entender. - -## Lista de Patrones de Diseño. - -### Patrones de Creación (Creational Patterns) - -Relativos al proceso de creación de un objeto. - -* [Abstract Factory](#abstract-factory) -* [Builder](#builder) -* [Factory](#factory) -* [Prototype](#prototype) -* [Singleton](#singleton) - -### Patrones de Estructura (Structural Patterns) - -Composición de clases u objetos. - -* [Adapter](#adapter) -* [Composite](#composite) -* [Decorator](#decorator) -* [Facade](#facade) -* [Proxy](#proxy) - -### Patrones de Comportamiento (Behavioral Patterns) - -Forma en que clases las u objetos interaccionan y distribuyen funcionalidades. - -* [Command](#command) -* [Observer](#observer) -* [Strategy](#strategy) - -## Strategy [↑](#lista-de-patrones) - -**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:** -* [Robot](https://github.com/LuisBurgos/design-patterns/tree/master/src/strategy/examples/robot) -* [Modos de transportación](https://github.com/LuisBurgos/design-patterns/tree/master/src/strategy/examples/transportation) - -## Observer [↑](#lista-de-patrones) - -**Propósito:** Defina 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](http://docs.oracle.com/javase/8/docs/api/java/util/Observer.html) - -**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. - -* [Subasta](https://github.com/LuisBurgos/design-patterns/tree/master/src/observer/examples/auction) - -## Factory [↑](#lista-de-patrones) - -**Propósito:** Definir una interface para crear un objeto, dejando a las subclases decidir de que tipo de clase se realizará la instancia. 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 este solo 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:** - -* [Carros](https://github.com/LuisBurgos/design-patterns/tree/master/src/factory/examples/cars) -* [Naves](https://github.com/LuisBurgos/design-patterns/tree/master/src/factory/examples/ships) - -## Abstract Factory [↑](#lista-de-patrones) - -**Propósito:** Proveer una interfaz para la creación de familias o 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:** - -* [Ovnis](https://github.com/LuisBurgos/design-patterns/tree/master/src/abstractfactory/examples/ufo) -* [El Reino](https://github.com/LuisBurgos/design-patterns/tree/master/src/abstractfactory/examples/kingdom) - -## Composite [↑](#lista-de-patrones) - -**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:** -* [Menu](https://github.com/LuisBurgos/design-patterns/tree/master/src/composite/examples/menu) -* [Sistema de Archivos](https://github.com/LuisBurgos/design-patterns/tree/master/src/composite/examples/directories) -* [Cartas](https://github.com/iluwatar/java-design-patterns/tree/master/composite/src/main/java/com/iluwatar/composite) (Externo). - -## Singleton [↑](#lista-de-patrones) - -**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:** -* [Gobierno](https://github.com/LuisBurgos/design-patterns/tree/master/src/singleton/examples/government) - -## Builder [↑](#lista-de-patrones) - -**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 muchoso 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:** -* [Heroes](https://github.com/LuisBurgos/design-patterns/tree/master/src/builder/examples/heroes) (basado en el siguiente [proyecto](https://github.com/iluwatar/java-design-patterns/tree/master/builder/src/main/java/com/iluwatar/builder)) -* [Pizza](https://github.com/LuisBurgos/design-patterns/tree/master/src/builder/examples/pizza) -* [Robots](https://github.com/LuisBurgos/design-patterns/tree/master/src/builder/examples/robots) -* [Comida rápida](https://github.com/LuisBurgos/design-patterns/tree/master/src/builder/examples/fastfood) - -## Prototype [↑](#lista-de-patrones) - -**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:** -* [Animales](https://github.com/LuisBurgos/design-patterns/tree/master/src/prototype/examples/animals) - -## Adapter [↑](#lista-de-patrones) - -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...](https://github.com/LuisBurgos/design-patterns/tree/master/src/adapter/pattern) -* 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:** -* [Books](https://github.com/LuisBurgos/design-patterns/tree/master/src/adapter/examples/books) - -## Decorator [↑](#lista-de-patrones) - -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...](https://github.com/LuisBurgos/design-patterns/tree/master/src/decorator/pattern) -* 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 ncesitemos 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:** -* [Pizzas](https://github.com/LuisBurgos/design-patterns/tree/master/src/decorator/examples/pizzas) - -## Facade [↑](#lista-de-patrones) - -**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...](https://github.com/LuisBurgos/design-patterns/tree/master/src/facade/pattern) -* 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:** -* [Banco](https://github.com/LuisBurgos/design-patterns/tree/master/src/facade/examples/bank) -* [Computadora](https://github.com/LuisBurgos/design-patterns/tree/master/src/facade/examples/computer) - -## Command [↑](#lista-de-patrones) - - 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...](https://github.com/LuisBurgos/design-patterns/tree/master/src/command/pattern) -* 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:** -* [Electrónicos](https://github.com/LuisBurgos/design-patterns/tree/master/src/command/examples/devices) -* [Hechizos](https://github.com/LuisBurgos/design-patterns/tree/master/src/command/examples/spells) - -## Proxy [↑](#lista-de-patrones) - -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. - -**Aplicación:** Usamos el patrón [Proxy...](https://github.com/LuisBurgos/design-patterns/tree/master/src/proxy/pattern) -* 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:** -* [Images](https://github.com/LuisBurgos/design-patterns/tree/master/src/proxy/examples/images) -* [ATM](https://github.com/LuisBurgos/design-patterns/tree/master/src/proxy/examples/atm) +# 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 descripcipnes breves y con aplicaciones en ejemplos muy sencillos de entender. + +## Lista de Patrones de Diseño. + +Ámbito: + +* [**C**] Clase +* [**O**] Objeto + +### Patrones de Creación (Creational Patterns) + +Relativos al proceso de creación de un objeto. + +* [Abstract Factory](#abstract-factory) [**O**] +* [Builder](#builder) [**O**] +* [Factory o Factory Method](#factory) [**C**] +* [Prototype](#prototype) [**O**] +* [Singleton](#singleton) [**O**] +* [Object Pool](#objectpool) [**O**] + +### Patrones de Estructura (Structural Patterns) + +Composición de clases u objetos. + +* [Adapter](#adapter) [**O**] [**C**] +* [Composite](#composite) [**O**] +* [Decorator](#decorator) [**O**] +* [Facade](#facade) [**O**] +* [Proxy](#proxy) [**O**] +* [Bridge] [**O**] +* [Flyweight] [**O**] + +### Patrones de Comportamiento (Behavioral Patterns) + +Forma en que las clases u objetos interaccionan y distribuyen funcionalidades. + +* [Command](#command) [**O**] +* [Observer](#observer) [**O**] +* [Strategy](#strategy) [**O**] +* [Chain of Responsability] [**O**] +* [Memento] [**O**] +* [Mediator] [**O**] +* [Templete method] [**C**] +* [Iterator] [**O**] +* [Visitor] [**O**] +* [State] [**O**] +* [Interpreter] [**C**] + +## Strategy [↑](#lista-de-patrones) + +**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:** +* [Robot](https://github.com/LuisBurgos/design-patterns/tree/master/src/strategy/examples/robot) +* [Modos de transportación](https://github.com/LuisBurgos/design-patterns/tree/master/src/strategy/examples/transportation) + +## Observer [↑](#lista-de-patrones) + +**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](http://docs.oracle.com/javase/8/docs/api/java/util/Observer.html) + +**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. + +* [Subasta](https://github.com/LuisBurgos/design-patterns/tree/master/src/observer/examples/auction) + +## Factory o Factory Method [↑](#lista-de-patrones) + +**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:** + +* [Carros](https://github.com/LuisBurgos/design-patterns/tree/master/src/factory/examples/cars) +* [Naves](https://github.com/LuisBurgos/design-patterns/tree/master/src/factory/examples/ships) + +## Abstract Factory [↑](#lista-de-patrones) + +**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:** + +* [Ovnis](https://github.com/LuisBurgos/design-patterns/tree/master/src/abstractfactory/examples/ufo) +* [El Reino](https://github.com/LuisBurgos/design-patterns/tree/master/src/abstractfactory/examples/kingdom) + +## Composite [↑](#lista-de-patrones) + +**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:** +* [Menu](https://github.com/LuisBurgos/design-patterns/tree/master/src/composite/examples/menu) +* [Sistema de Archivos](https://github.com/LuisBurgos/design-patterns/tree/master/src/composite/examples/directories) +* [Cartas](https://github.com/iluwatar/java-design-patterns/tree/master/composite/src/main/java/com/iluwatar/composite) (Externo). + +## Singleton [↑](#lista-de-patrones) + +**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:** +* [Gobierno](https://github.com/LuisBurgos/design-patterns/tree/master/src/singleton/examples/government) + +## Builder [↑](#lista-de-patrones) + +**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:** +* [Heroes](https://github.com/LuisBurgos/design-patterns/tree/master/src/builder/examples/heroes) (basado en el siguiente [proyecto](https://github.com/iluwatar/java-design-patterns/tree/master/builder/src/main/java/com/iluwatar/builder)) +* [Pizza](https://github.com/LuisBurgos/design-patterns/tree/master/src/builder/examples/pizza) +* [Robots](https://github.com/LuisBurgos/design-patterns/tree/master/src/builder/examples/robots) +* [Comida rápida](https://github.com/LuisBurgos/design-patterns/tree/master/src/builder/examples/fastfood) + +## Prototype [↑](#lista-de-patrones) + +**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:** +* [Animales](https://github.com/LuisBurgos/design-patterns/tree/master/src/prototype/examples/animals) + +## Adapter [↑](#lista-de-patrones) + +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...](https://github.com/LuisBurgos/design-patterns/tree/master/src/adapter/pattern) +* 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:** +* [Books](https://github.com/LuisBurgos/design-patterns/tree/master/src/adapter/examples/books) + +## Decorator [↑](#lista-de-patrones) + +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...](https://github.com/LuisBurgos/design-patterns/tree/master/src/decorator/pattern) +* 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 ncesitemos 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:** +* [Pizzas](https://github.com/LuisBurgos/design-patterns/tree/master/src/decorator/examples/pizzas) + +## Facade [↑](#lista-de-patrones) + +**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...](https://github.com/LuisBurgos/design-patterns/tree/master/src/facade/pattern) +* 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:** +* [Banco](https://github.com/LuisBurgos/design-patterns/tree/master/src/facade/examples/bank) +* [Computadora](https://github.com/LuisBurgos/design-patterns/tree/master/src/facade/examples/computer) + +## Command [↑](#lista-de-patrones) + + 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...](https://github.com/LuisBurgos/design-patterns/tree/master/src/command/pattern) +* 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:** +* [Electrónicos](https://github.com/LuisBurgos/design-patterns/tree/master/src/command/examples/devices) +* [Hechizos](https://github.com/LuisBurgos/design-patterns/tree/master/src/command/examples/spells) + +## Proxy [↑](#lista-de-patrones) + +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. + +**Aplicación:** Usamos el patrón [Proxy...](https://github.com/LuisBurgos/design-patterns/tree/master/src/proxy/pattern) +* 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:** +* [Images](https://github.com/LuisBurgos/design-patterns/tree/master/src/proxy/examples/images) +* [ATM](https://github.com/LuisBurgos/design-patterns/tree/master/src/proxy/examples/atm) + +## Object Pool [↑](#lista-de-patrones) + +**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. + +**Ejemplos:** +* [Ejemplo](https://github.com/peterm85/design-patterns/tree/master/src/objectpool/pattern) + +**Referencia:** https://www.javatpoint.com/object-pool-pattern diff --git a/src/objectpool/pattern/Client.java b/src/objectpool/pattern/Client.java new file mode 100644 index 0000000..a57074b --- /dev/null +++ b/src/objectpool/pattern/Client.java @@ -0,0 +1,29 @@ +package objectpool.pattern; + +public class Client implements Runnable { + + private ObjectPool pool; + private int threadNo; + + public Client(ObjectPool pool, int threadNo){ + this.pool = pool; + this.threadNo = threadNo; + } + + public void run() { + // get an object from the pool + HeavyObject heavyObject = pool.borrowObject(); + System.out.println("Thread " + threadNo + ": Object with process no. " + heavyObject.getObjectNo() + " was borrowed"); + + try { + heavyObject.doStuff(); + } catch (InterruptedException e) { + e.printStackTrace(); + } + + // return ExportingProcess instance back to the pool + pool.returnObject(heavyObject); + + System.out.println("Thread " + threadNo +": Object with process no. " + heavyObject.getObjectNo() + " was returned"); + } +} diff --git a/src/objectpool/pattern/HeavyObject.java b/src/objectpool/pattern/HeavyObject.java new file mode 100644 index 0000000..491a25b --- /dev/null +++ b/src/objectpool/pattern/HeavyObject.java @@ -0,0 +1,26 @@ +package objectpool.pattern; + +import java.util.concurrent.TimeUnit; + +public class HeavyObject { + + private long objectNo; + + public HeavyObject(long objectNo) { + this.objectNo = objectNo; + + // do some expensive calls / tasks here in future + // ......... + + System.out.println("Object with no. " + objectNo + " was created"); + } + + public long getObjectNo() { + return objectNo; + } + + public void doStuff() throws InterruptedException { + //TimeUnit.SECONDS.sleep(1); + System.out.println("Object with no. " + objectNo + " is doing something"); + } +} diff --git a/src/objectpool/pattern/ObjectPool.java b/src/objectpool/pattern/ObjectPool.java new file mode 100644 index 0000000..c7e1eed --- /dev/null +++ b/src/objectpool/pattern/ObjectPool.java @@ -0,0 +1,32 @@ +package objectpool.pattern; + +import java.util.concurrent.ConcurrentLinkedQueue; + +abstract class ObjectPool { + + private ConcurrentLinkedQueue pool; + + public ObjectPool(final int minObjects) { + pool = new ConcurrentLinkedQueue(); + for (int i = 0; i < minObjects; i++) { + pool.add(createObject()); + } + } + + protected abstract T createObject(); + + public T borrowObject() { + T object; + if ((object = pool.poll()) == null) { + object = createObject(); + } + return object; + } + + public void returnObject(T object) { + if (object == null) { + return; + } + this.pool.offer(object); + } +} diff --git a/src/objectpool/pattern/ObjectPoolTest.java b/src/objectpool/pattern/ObjectPoolTest.java new file mode 100644 index 0000000..1c495ff --- /dev/null +++ b/src/objectpool/pattern/ObjectPoolTest.java @@ -0,0 +1,49 @@ +package objectpool.pattern; + +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.concurrent.TimeUnit; +import java.util.concurrent.atomic.AtomicLong; + +public class ObjectPoolTest { + + private ObjectPool pool; + private AtomicLong processNo=new AtomicLong(0); + + public static void main(String args[]) { + ObjectPoolTest op=new ObjectPoolTest(); + op.setUp(); + op.testObjectPool(); + } + + public void setUp() { + // Create a pool of objects of type HeavyObject. + pool = new ObjectPool(4){ + protected HeavyObject createObject() { + // create a test object which takes some time for creation + return new HeavyObject( processNo.incrementAndGet() ); + } + }; + } + + public void testObjectPool() { + ExecutorService executor = Executors.newFixedThreadPool(8); + + // execute 8 tasks in separate threads + executor.execute(new Client(pool, 1)); + executor.execute(new Client(pool, 2)); + executor.execute(new Client(pool, 3)); + executor.execute(new Client(pool, 4)); + executor.execute(new Client(pool, 5)); + executor.execute(new Client(pool, 6)); + executor.execute(new Client(pool, 7)); + executor.execute(new Client(pool, 8)); + + executor.shutdown(); + try { + executor.awaitTermination(30, TimeUnit.SECONDS); + } catch (InterruptedException e){ + e.printStackTrace(); + } + } +}