- Composite (patrón de diseño)
-
El patrón Composite sirve para construir objetos complejos a partir de otros más simples y similares entre sí, gracias a la composición recursiva y a una estructura en forma de árbol.
Esto simplifica el tratamiento de los objetos creados, ya que al poseer todos ellos una interfaz común, se tratan todos de la misma manera.
Contenido
Problema que soluciona
Imaginemos que necesitamos crear una serie de clases para guardar información acerca de una serie de figuras que serán círculos, cuadrados y triángulos. Además necesitamos poder tratar también grupos de imágenes porque nuestro programa permite seleccionar varias de estas figuras a la vez para moverlas por la pantalla.
En principio tenemos las clases Círculo, Cuadrado y Triángulo, que heredarán de una clase padre que podríamos llamar Figura e implementarán todas la operación pintar(). En cuanto a los grupos de Figuras podríamos caer en la tentación de crear una clase particular separada de las anteriores llamada GrupoDeImágenes, también con un método pintar().
Problema.
Esta idea de separar en clases privadas componentes (figuras) y contenedores (grupos) tiene el problema de que, para cada uno de los dos atributos, el método pintar() tendrá una implementación diferente, aumentando la complejidad del sistema.
Implementación
El patrón Composite da una solución elegante a este problema, de la que además resulta en una implementación más sencilla.
A la clase Figura la llamaríamos Gráfico y de ella extenderían tanto Círculo, Cuadrado y Triángulo, como GrupoDeImágenes. Además, ésta última tendría una relación todo-parte de multiplicidad * con Gráfico: un GrupoDeImágenes contendría varios Gráficos, ya fuesen éstos Cuadrados, Triángulos, u otras clases GrupoDeImágenes.
Así, es posible definir a un grupo de imágenes recursivamente. Por ejemplo, un objeto cuya clase es GrupoDeImágenes podría contener un Cuadrado, un Triángulo y otro GrupoDeImágenes, este grupo de imágenes podría contener un Círculo y un Cuadrado. Posteriormente, a este último grupo se le podría añadir otro GrupoDeImágenes, generando una estructura de composición recursiva en árbol, por medio de muy poca codificación y un diagrama sencillo y claro.
Diagrama
Ejemplos de utilización
En Java: las clases java.awt.Component (Componente), java.awt.Container (Contenedor), java.awt.Panel (Contenedor concreto), java.awt.Button (Boton)
Código en Java
import java.util.*; public abstract class Componente { protected String nombre; public Componente (String nombre) { this.nombre = nombre; } abstract public void Agregar(Componente c); abstract public void Eliminar(Componente c); abstract public void Mostrar(int profundidad); } class Compuesto extends Componente { private ArrayList<Componente> hijo = new ArrayList<Componente>(); public Compuesto (String name) { super(name); } @Override public void Agregar(Componente componente) { hijo.add(componente); } @Override public void Eliminar(Componente componente) { hijo.remove(componente); } @Override public void Mostrar(int profundidad) { System.out.println(nombre + " nivel: " + profundidad); for (int i = 0; i < hijo.size(); i++) hijo.get(i).Mostrar(profundidad + 1); } } class Hoja extends Componente { public Hoja (String nombre) { super(nombre); } public void Agregar(Componente c) { System.out.println("no se puede agregar la hoja"); } public void Eliminar(Componente c) { System.out.println("no se puede quitar la hoja"); } public void Mostrar(int depth) { System.out.println('-' + "" + nombre); } } public class Client { public static void main(String[] args) { Compuesto raiz = new Compuesto("root"); raiz.Agregar(new Hoja("hoja A")); raiz.Agregar(new Hoja("hoja B")); Compuesto comp = new Compuesto("compuesto X"); comp.Agregar(new Hoja("hoja XA")); comp.Agregar(new Hoja("hoja XB")); raiz.Agregar(comp); raiz.Agregar(new Hoja("hoja C")); Hoja l = new Hoja("hoja D"); raiz.Agregar(l); raiz.Eliminar(l); raiz.Mostrar(1); } }
Categoría:- Patrones de diseño
Wikimedia foundation. 2010.