桥接模式

Posted 每天一篇,坚持...

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了桥接模式相关的知识,希望对你有一定的参考价值。

场景:

  商场系统中常见的商品分类,以电脑为,如何良好的处理商品分类销售的问题 ?

  

 

  我们可以使用多重继承实现上图的关系

 问题:

  扩展性问题(类个数膨胀问题:)

    如果要增加一个新的电脑类型:智能手机,则要增加各个品牌下面的类

    如果要增加一个新的品牌,也要增加各种电脑类型的类

  违反单一职责原则:

    一个类:联想笔记本,有两个引起这个类变化的原因

 

首先看一个常见的例子: 
未使用桥接模式的类图如下:

package com.note.pattern.bridge;

public interface Computer {
	void sale();
}

class Desktop implements Computer{
	@Override
	public void sale() {
		System.out.println("销售台式机");
	}
}

class Laptop implements Computer{
	@Override
	public void sale() {
		System.out.println("销售笔记本!");
	}
}

class Pad implements Computer {
    @Override
    public void sale() {
        System.out.println("销售平板电脑!");
    }
}

class LenovoDesktop extends Desktop {
    @Override
    public void sale() {
        System.out.println("销售联想台式机");
    }
}

class LenovoLaptop extends Laptop {
    @Override
    public void sale() {
        System.out.println("销售联想笔记本");
    }
}

class LenovoPad extends Pad {
    @Override
    public void sale() {
        System.out.println("销售联想平板电脑");
    }
}


class ShenzhouDesktop extends Desktop {
    @Override
    public void sale() {
        System.out.println("销售神舟台式机");
    }
}
class ShenzhouLaptop extends Laptop {
    @Override
    public void sale() {
        System.out.println("销售神舟笔记本");
    }
}
class ShenzhouPad extends Pad {
    @Override
    public void sale() {
        System.out.println("销售神舟平板电脑");
    }
}

class DellDesktop extends Desktop {
    @Override
    public void sale() {
        System.out.println("销售戴尔台式机");
    }
}
class DellLaptop extends Laptop {
    @Override
    public void sale() {
        System.out.println("销售戴尔笔记本");
    }
}
class DellPad extends Pad {
    @Override
    public void sale() {
        System.out.println("销售戴尔平板电脑");
    }
}

  可以发现,这种类之间的关系的设计是存在扩展性问题的,如果要增加一个新的电脑类型–智能手机,则要增加各个品牌下面的类;如果要增加一个新的品牌,也要增加各种电脑类型的类。 
  这种设计违反了单一职责原则:一个类,却有两个引起该类变化的原因。

  

  如果遇到了有两种或者两种以上变化维度的场景,适合使用桥接模式,将各个维度连接起来。 
  上面的例子使用桥接模式的类图如下,我们可以将Computer类作为不同类型电脑的一个维度;创建一个Brand类,作为不同品牌的一个维度,最后将这两个维度的类通过组合方式桥接到一起:

  

package com.note.pattern.bridge;

/**
 * 品牌 
 */
public interface Brand {
	
	void sale();
	
}


class Lenovo implements Brand {

    @Override
    public void sale() {
        System.out.println("销售联想电脑");
    }

}

class Dell implements Brand {

    @Override
    public void sale() {
        System.out.println("销售Dell电脑");
    }

}

  

package com.note.pattern.bridge;

public class Computer2 {
	
	protected Brand mBrand;

	public Computer2(Brand mBrand) {
		this.mBrand = mBrand;
	}
	
	public void sale() {
		mBrand.sale();
	}
	
}

class Desktop2 extends Computer2 {

    public Desktop2(Brand b) {
        super(b);
    }

    @Override
    public void sale() {
        super.sale();
        System.out.println("销售台式机");
    }
}


class Laptop2 extends Computer2 {

    public Laptop2(Brand b) {
        super(b);
    }

    @Override
    public void sale() {
        super.sale();
        System.out.println("销售笔记本");
    }
}

  

package com.note.pattern.bridge;

public class Client {

	public static void main(String[] args) {
		
		//销售联想笔记本电脑
		Computer2 computer2 = new Laptop2(new Lenovo());
		computer2.sale();
		
	}

}

  

桥接模式总结:

  桥接模式可以取代多层继承的方案。多层继承违背了单一职责原则,复用性较差,类的个数也非常多。桥接模式可以极大的减少子类的个数,从而降低管理和维护的成本。 
  桥接模式极大地提高了系统的可扩展性,在两个变化维度中任意扩展一个维度都不需要修改原有的系统,符合开闭原则。

  

 

以上是关于桥接模式的主要内容,如果未能解决你的问题,请参考以下文章

桥接模式 C++ 代码编译错误

案例分析:设计模式与代码的结构特性

桥接模式-代码实现

23种设计模式之桥接模式代码实例

设计模式之桥接模式

桥接模式