通常,在代码已经存在的情况下编写客户端代码(客户端就是需要调用我们代码的对象),开发人员可以采取模拟客户端的方式调用我们提供的接口对象。然而,客户端代码也可能与你的代码单独进行开发,这种情况下,会发现虽然现有代码提供了客户端需要的服务,但被定义成不同的方法名。这时,我们就需要采用适配器模式。适配器模式的意图在于,使用不同接口的类所提供的服务为客户端提供它所需要的接口。
适配器模式可分为两种:
(1)类适配器
(2)对象适配器
类适配器:
当我们适配现有代码时,发现客户端开发人员已经考虑到了这种情形,开发人员为客户端使用的服务提供了接口定义。如下:
public interface RequiredInterface { public void requiredMethod(); }
RequiredInterface接口定义了客户端需要调用的服务方法requiredMethod,在ExsitingClass类中,则定义了usefulMethod方法,它是客户端需要的服务实现。
public class ExistingClass { public void usefulMethod() { System.out.println("useful method!!!"); } }
若要对ExistingClass进行适配,满足客户端需要,则可以创建一个NewClass类,继承ExistingClass,同时继承RequiredInterface接口,通过重写requiredMethod方法将客户端的请求委托给usefulMethod方法,如下:
public class NewClass extends ExistingClass implements RequiredInterface{ public void requiredMethod() { this.usefulMethod(); } }
图1
图1中的NewClass类就是类适配器的例子。该类的实例同时也是RequiredInterface的实例。换言之,NewClass类满足类客户端的需求。
当客户端在接口中定义了它所期待当行为时,就可以使用类适配器,提供一个实现该接口的类。倘若没有定义客户端的接口,则不能使用类适配器,此时必须使用对象适配器。在类适配器中,新的适配器类实现了定义的接口,同时继承现有类。当你需要适配的方法并非定义在接口中时,这种方法就不凑效类。此时我们可以创建一个对象适配器,使用委托而非继承,如下图所示:
上图中的NewClass是对象适配器的一个例子,该类的实例也是RequiredClass的实例,换言之,NewClass满足了客户端的需要。NewClass通过使用ExistingClass实例对象,可以将ExistingClass类适配为符合客户端的需要。
public class RequiredClass { public void requiredMethod(){ } } public class NewClass extends RequiredClass { private ExistingClass existingClass = new ExistingClass(); @Override public void requiredMethod() { existingClass.usefulMethod(); } }
适配器模式可以是我们重用一个现有的类,以满足客户端的需要。当客户端通过接口表达其需求时,采用类适配器,创建一个实现该接口的类,同时继承现有类。 当客户端没有指定它所需要的接口时,应采用对象适配器,创建一个类使用现有类的实例。如果我们不需要(或者不能)重写客户端可能调用的方法时,这种方式可能存在一定的危险性。