1、桥接模式的核心要点:
- 处理多继承结构,处理多维度变化的场景,将各个维度设计成独立的继承结构,使各个维度可以独立的扩展,并且在抽象层建立关联。
2、举例说明,商城系统中常见的商品分类,以电脑为例,如何良好的处理商品分类的销售的问题呢?
- 在没有使用桥接模式时,我们能想到的方法是使用多层继承结构实现,以下为实现的代码:
package com.geeklicreed.bridge; public interface Computer { void sale(); } class Desktop implements Computer{ public void sale(){ System.out.println("销售台式机"); } } class Laptop implements Computer{ public void sale(){ System.out.println("销售笔记本"); } } class Pad implements Computer{ public void sale(){ System.out.println("销售平板电脑"); } } class LenovoDesktop extends Desktop{ @Override public void sale() { super.sale(); System.out.println("销售联想台式机"); } } class DellDesktop extends Desktop{ @Override public void sale() { super.sale(); System.out.println("销售戴尔台式机"); } } class HaseeDesktop extends Desktop{ @Override public void sale() { super.sale(); System.out.println("销售神舟台式机"); } } class LenovoLaptop extends Laptop{ public void sale(){ super.sale(); System.out.println("销售联想笔记本"); } } class DellLaptop extends Laptop{ public void sale(){ super.sale(); System.out.println("销售戴尔笔记本"); } } class HaseeLaptop extends Laptop{ public void sale(){ super.sale(); System.out.println("销售神舟笔记本"); } } class LenovoPad extends Pad{ public void sale(){ super.sale(); System.out.println("销售联想平板电脑"); } } class DellPad extends Pad{ public void sale(){ super.sale(); System.out.println("销售戴尔平板电脑"); } } class HaseePad extends Pad{ public void sale(){ super.sale(); System.out.println("销售神舟平板电脑"); } }
- 扩展性问题(类个数膨胀问题):
- 如果要增加一个新的电脑类型:智能手机,则要增加各个品牌下面的类。
- 如果要增加一个新的品牌,则要增加各种电脑类型的类。(原因在于品牌和电脑类型是两个不同的维度,而 java 只能实现单继承。)
3、使用桥接模式解决以上出现的问题,我们来看看示例的代码:
a、可以看到,先创建了一个 Brand 的接口,后 Lenovo、Dell、Hasee 类实现该接口。再创建一个 Computer2 类,Computer2 类和 Brand 接口是组合的关系。(即在 Computer2 类里面定义了一个 Brand 类的对象实例)
package com.geeklicreed.bridge; public class Computer2 { protected Brand brand; public Computer2(Brand brand) { this.brand = brand; } public void sale() { System.out.println("销售电脑类型:"); } } class Desktop extends Computer2 { public Desktop(Brand brand) { super(brand); } public void sale() { super.sale(); System.out.println("销售台式机"); } } class PadType extends Computer2 { public PadType(Brand brand) { super(brand); } public void sale() { super.sale(); System.out.println("销售平板电脑"); } } class Laptop extends Computer2 { public Laptop(Brand brand) { super(brand); } public void sale() { super.sale(); System.out.println("销售笔记本"); } }
package com.geeklicreed.bridge; public interface Brand { void sale(); } class Lenovo implements Brand{ public void sale(){ System.out.println("销售联想品牌"); } } class Dell implements Brand{ public void sale(){ System.out.println("销售戴尔品牌"); } } class Hasee implements Brand{ public void sale(){ System.out.println("销售神舟品牌"); } }
- 注:可以看到,增加新的品牌或者是新的电脑类型,只需要加入一个新类就可以了,这样程序的扩展性比较好。
b、在 Client09 测试类下测试,并且画出示例中的类图关系:
package com.geeklicreed.bridge; public class Client { public static void main(String[] args) { Computer2 computer2 = new Desktop(new Hasee()); computer2.brand.sale(); computer2.sale(); /* 输出结果为: 销售神舟品牌 销售电脑类型: 销售台式机*/ } }
4、桥接模式总结:
- 桥接模式可以取代多层继承的方案。多层继承违背了单一职责原则,复用性较差,类的个数也非常多。桥接模式可以极大的减少子类的个数,从而降低管理和维护的成本。
- 桥接模式极大的提高了系统可扩展性,在两个变化维度中任意扩展一个维度,都不需要修改原有的系统,符合开闭原则。