Convierte la interfaz de una clase en otra interfaz que el cliente espera. El adaptador permite a las clases trabajar juntas, lo que de otra manera no podrÃa hacerse debido a sus interfaces incompatibles.
Es recomendable utilizar el patrón adaptador cuando:
se desea usar una clase existente, y su interfaz no sea igual a la necesitada.
cuando se desea crear una clase reutilizable que coopere con clases no relacionadas. Es decir, que las clases no tienen necesariamente interfaces compatibles.
Estructura
Participantes
Target define la interfaz especÃfica del dominio que Client usa.
Client colabora con la conformación de objetos para la interfaz Target.
Adaptee define una interfaz existente que necesita adaptarse.
Adapter adapta la interfaz de Adaptee a la interfaz Target.
Colaboraciones
Client llama a las operaciones sobre una instancia Adapter. De hecho, el adaptador llama a las operaciones de Adaptee que llevan a cabo el pedido.
Consecuencias
Los adaptadores de clase y objetos tienen varios pros y contras.
Un adaptador de clase:
adapta Adaptee a Target encargando a una clase Adaptee concreta. Como consecuencia, una clase adaptadora no funcionará cuando se desea adaptar una clase y todas sus subclases.
permite a los Adapter sobreescribir algo de comportamiento de Adaptee, ya que Adapter es una subclase de Adaptee.
hace difÃcil sobrescribir el comportamiento de Adaptee. Esto requerirá derivar Adaptee y hacer que Adapter se refiera a la subclase en lugar que al Adaptee por sà mismo.
Aquà hay otras cuestiones a considerar cuando se utiliza el patrón Adapter:
¿Cuanta adaptación hace el Adapter?Adapter varÃa en la cantidad de trabajo que hace para adaptar Adaptee a la interfaz Target. Hay un espectro de trabajo posible, desde una simple conversión (por ejemplo, cambiando los nombres de las operaciones) hasta soportando un conjunto de operaciones enteramente diferentes. La cantidad de trabajo que Adapter hace depende de cuanto de similar tienen la interfaz Target con Adaptee.
Adaptadores enganchables. Una clase es más reutilizable cuando se deja a un lado la suposición de que otras clases deben utilizarla. Convirtiendo la adaptación de una interfaz en una clase, se elimina la suposición de que otras clases ven la misma interfaz. Dicho de otra manera, la adaptación de la interfaz permite incorporar a la clase en sistemas existentes que pueden esperar diferentes interfaces de la misma.
Implementación
Crear una nueva clase que será el Adaptador, que extienda del componente existente e implemente la interfaz obligatoria. De este modo se tiene la funcionalidad que se querÃa y se cumple la condición de implementar la interfaz.
La diferencia entre los patrones adaptador y fachada (facade) es que el primero reutiliza una interfaz ya existente, mientras que el segundo define una nueva con el objetivo de simplificarla.
packageStructural_patterns;publicclassAdapterWrapperPattern{/** * Client: Uses (and only knows) Target Interface */publicstaticvoidmain(Stringargs[]){GuitareGuitar=newElectricGuitar();eGuitar.onGuitar();eGuitar.offGuitar();GuitareAGuitar=newElectricAcousticGuitar();eAGuitar.onGuitar();eAGuitar.offGuitar();}/** * Target: Class used by the Client */publicinterfaceGuitar{publicvoidonGuitar();publicvoidoffGuitar();}/** * Direct Target Implementation: Already Adapted/Wrapped */publicclassElectricGuitarimplementsGuitar{publicvoidonGuitar(){System.out.println("Playing Guitar");}publicvoidoffGuitar(){System.out.println("I'm tired to play the guitar");}}/** * Adaptee: Class to Adapter/Wrapper */publicclassAcousticGuitar{publicvoidplay(){System.out.println("Playing Guitar");}publicvoidleaveGuitar(){System.out.println("I'm tired to play the guitar");}}/** * Adapter/Wrapper: We Adapter/Wrapper AcousticGuitar into * ElectricAcousticGuitar to adapt into the Guitar Model */publicclassElectricAcousticGuitarimplementsGuitar{AcousticGuitaracoustic=newAcousticGuitar();publicvoidonGuitar(){acoustic.play();}publicvoidoffGuitar(){acoustic.leaveGuitar();}}}