桥接模式:抽象与现实分离,使各自可以独立变化
demo描述:手机具有形状和品牌两个属性,要不用品牌的手机有不同的形状。如果采用继承方式实现,要么品牌继承形状导致品牌冗余、要么形状继承品牌导致形状冗余,用数学里的排列组合的方式来抽离出形状与品牌两个属性,让其组合,这样既能保证不冗余又能保证手机种类齐全。这里就是将属性形状、品牌搭建了个桥,做了个桥接模式。
demo代码:
品牌接口,定义手机应具备的功能;实现类,实现接口的功能:
public interface Brand {
void open();
void close();
void call();
}
//---------------------------------------------------------
public class Vivo implements Brand {
@Override
public void open() {System.out.println("vivo手机开机");}
@Override
public void close() {System.out.println("vivo手机关机");}
@Override
public void call() {System.out.println("vivo手机打电话");}
}
public class XiaoMi implements Brand {
@Override
public void open() {System.out.println("小米手机开机");}
@Override
public void close() {System.out.println("小米手机关机");}
@Override
public void call() {System.out.println("小米手机打电话");}
}
手机抽象类,组合品牌(手机的属性),拥有品牌下的方法;手机具体类,定义手机的样式(手机的属性):
public abstract class Phone {
//组合品牌
private Brand brand;
public Phone(Brand brand) {this.brand = brand;}
protected void open() {this.brand.open();}
protected void close() {this.brand.close();}
protected void call() {this.brand.call();}
}
//--------------------------------------------------
public class FoldedPhone extends Phone {
public FoldedPhone(Brand brand) {
super(brand);
}
@Override
protected void open() {
super.open();
System.out.println("折叠样式手机");
}
@Override
protected void close() {
super.close();
System.out.println("折叠样式手机");
}
@Override
protected void call() {
super.call();
System.out.println("折叠样式手机");
}
}
public class UpRightPhone extends Phone {
public UpRightPhone(Brand brand) {
super(brand);
}
@Override
protected void open() {
super.open();
System.out.println("直立式手机");
}
@Override
protected void close() {
super.close();
System.out.println("直立式手机");
}
@Override
protected void call() {
super.call();
System.out.println("直立式手机");
}
}
客户端:
public class Client1 {
public static void main(String[] args) {
FoldedPhone phone1 = new FoldedPhone(new XiaoMi());
phone1.open();
phone1.call();
phone1.close();
}
}
demo类图:
类图分析:将手机的品牌、形状两个属性分离,一个属性(品牌)作为接口定义手机的功能,一个属性以抽象类的方式绑定具体的类调用接口的功能,以品牌*形状的方式联合描述手机。图中聚合关系相当于桥梁,连接起两个属性,让其可以双向扩展,任意一方或双方的扩展都仅仅是新增扩展出的类即可。
如果是品牌+属性这种基础方式,则扩展时要增加一个类系,不香。
适用场景:类具有二维属性,且属性有联合描述类的特征时可用桥接模式拆分属性,通过搭桥的方式实现属性描述类的降维打击
总结:桥接模式将具有二维属性的类的属性组合延迟到了使用时,而不是类产生时就将属性组合好,这样做可以减少属性+属性这种方式导致的冗余,不会因为继承或多层次继承而导致系统类个数剧增。