前面三篇讲述了java设计模式中的创建类型,主要关注点在于类创建对象时有哪些模式可以参照,如:简单工厂、工厂方法、抽象工厂、单例、建造者模式这几种比较优雅的类创建出对象的设计模式。
下面我们将关注点放在如何处理类与类之间的关系上,让类与类之间的结构、以及协作处理变得更加的优雅。
在java中,让类与类之间发生关系有两种方式:一是继承,另一个是组合。结构型设计模式关注焦点在于这两种方式中,倾向于组合,会优先通过组合的方式,让类之间产生关系;而少使用继承。
下面看下这两种关系:
图中的target类,要是想要获取A类或者B类中的功能,可以通过继承A类或者B类;也可以在Target类中new出A类和B类的对象,这种就是将A类和B类组合到了target类中去使用。为什么要优先使用组合的方式呢?因为在组合中new出来的A类我们是持有的引用,可以通过get、set方法获取、控制A类的对象和方法,也可以获取到A类的子类对象,这种方式计较灵活。
下面我们看下结构型设计模式中的--适配器设计模式。
这里我们继续讲Peter在水果工厂创业中遇到的问题,某天Peter发现水果店里面的桔子包装盒不够用了,但是有苹果、香蕉等其他的包装盒有多余,他就和顾客说明了下情况,先把桔子打包到苹果包装盒中,先运输到顾客手中即可,顾客也欣然答应了。
这里核心问题出现了,就是如何将苹果包装盒伪装成桔子包装盒呢?
下面我们就利用适配器模式,巧妙的将苹果的包装盒适配为桔子包装盒,软件编写的类图如下:
这个适配器就是OrangeBagAdapter,让它继承OrangeBag,重写pack方法,传入真正的调用者是AppleBag,这样就达到了“偷梁换柱”的效果,在外面看虽然是Orange的方法,但是内部却已经转化成了AppleBag的方法。
具体的实现:
public class OrangeBagAdapter extends OrangeBag {
private AppleBag appleBag;
public OrangeBagAdapter(AppleBag appleBag) {
this.appleBag = appleBag;
}
@Override
public void pack() {
appleBag.pack();
}
}
测试类:
public class AdapterClientTest {
public static void main(String[] args){
Orange orange = new Orange("peter",100);
OrangeBag bag = getBag2();
// orange.pack(bag);
bag.pack();
}
/**
* 取桔子包装
* @return
*/
private static OrangeBag getBag(){
OrangeBag bag = new OrangeBag();
return bag;
}
private static OrangeBag getBag2(){
//准备用途苹果盒代替
AppleBag appleBag = new AppleBag();
//把苹果盒适配成桔子包装盒
OrangeBag orangeBag = new OrangeBagAdapter(appleBag);
return orangeBag;
}
}
getBag()就是原始的桔子包装盒,getBag2()就是适配器的苹果包装盒:
最终成功的用苹果包装盒将桔子装下了,其实OrangeAdapter内部就是将苹果包装类组合进去,调用的是苹果包装方法,这就是组合方式,让两个类之间产生关系的适配器模式。