热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

[19/04/26-星期五]GOF23_结构型模式(桥接模式、组合模式)

一、桥接模式(bridge)场景:商城系统中常见的商品分类,以电脑为例,首先想到使用多层继承结构。——台式机(联想台式机、戴尔台式机、神舟台式机)电脑——笔记

一、桥接模式(bridge)

场景:商城系统中常见的商品分类,以电脑为例,首先想到使用多层继承结构。

          —— 台式机(联想台式机、戴尔台式机、神舟台式机)

电脑    ——笔记本(联想笔记本、戴尔笔记本、神舟笔记本)

           ——平板电脑(联想pad、戴尔pad、神舟pad)

问题:(1)如果要增加一个新的电脑类型,则要增加各个品牌下的类。

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

把另一个类的结构作为属性放在这个类中。

核心:处理多层继承结构,处理多维度变化的场景,将各个维度设计成独立的继承结构,使各个维度可以独立的扩展在抽象层建立关联

好处:桥接模式可以取代多层继承的方案。 多层继承违背了单一职责原则,复用性较差,类的个数也非常多。桥接模式可以极大的减少子类的个数,从而降低管理和维护的成本。

桥接模式极大的提高了系统可扩展性,在两个变化维度中任意扩展一个维度,都不需要修改原有的系统,符合开闭原则。

应用场景:

– JDBC驱动程序

– AWT中的Peer架构

– 银行日志管理:

• 格式分类:操作日志、交易日志、异常日志

• 距离分类:本地记录日志、异地记录日志

– 人力资源系统中的奖金计算模块:

• 奖金分类:个人奖金、团体奖金、激励奖金。

• 部门分类:人事部门、销售部门、研发部门。

– OA系统中的消息处理:

• 业务类型:普通消息、加急消息、特急消息

• 发送消息方式:系统内消息、手机短信、邮件

 

【不用桥接模式UML类图】

【代码】

/***
 * "电脑"接口 不用桥接模式
 */
package cn.sxt.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{
    public void sale() {
        System.out.println("销售联想台式机!");
    }
}
class LenovoLaptop extends Laptop{
    public void sale() {
        System.out.println("销售联想笔记本!");
    }
}
class LenovoPad extends Pad{
    public void sale() {
        System.out.println("销售联想平板!");
    }
}
//神舟系列
class ShenZhouDesktop extends Desktop{
    public void sale() {
        System.out.println("销售神舟台式机!");
    }
}
class ShenZhouLaptop extends Laptop{
    public void sale() {
        System.out.println("销售神舟笔记本!");
    }
}
class ShenZhouPad extends Pad{
    public void sale() {
        System.out.println("销售神舟平板!");
    }
}

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

 

【使用桥接模式UML类图】

【代码】

/***
 * "品牌"接口 和各种具体品牌类,又一个维度,2个维度相互独立,x轴
 */
package cn.sxt.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("销售戴尔牌电脑");
    }
}
//在这里,新加品牌“神舟”
class ShenZhou implements Brand{
    @Override
    public void sale() {
        System.out.println("销售神舟牌电脑");
    }
}

 

/**
 * 电脑类型,一个维度,y轴
 */
package cn.sxt.bridge;

public class Computer2 {
    protected Brand brand;//持有Brand类的引用,使电脑类Computer2 天然具有品牌类(Brand)的属性
    
    public Computer2(Brand b) {//构造器
        this.brand=b;
    }
    
    public void sale() {
        brand.sale();//调用的是Brand类中brand对象的sale方法
    }
}

class Desktop2 extends Computer2{
    public Desktop2 (Brand b) {
        super(b);
    }
    public void sale() {
        super.sale();
        System.out.println("销售台式机");    
    }
}
class Laptop2 extends Computer2{
    public Laptop2 (Brand b) {
        super(b);
    }
    public void sale() {
        super.sale();
        System.out.println("销售笔记本");    
    }
}

【客户】

/***
 * 客户端
 */
package cn.sxt.bridge;

public class Client {
    public static void main(String[] args) {
        //销售联想牌的笔记本电脑,用组合来代替继承,后期扩展很方便
        Computer2 c= new Laptop2(new Lenovo());
        c.sale();
        Computer2 c1= new Desktop2(new ShenZhou());
        c1.sale();
    }

}

 

二、组合模式(Composite、复合物)

场景:把部分和整体的关系用树形结构来表示,从而使客户端可以用统一的方式处理部分对象和整体对象。

核心:

– 抽象构件(Component)角色: 定义了叶子和容器构件的共同点

– 叶子(Leaf)构件角色:无子节点,叶子节点。

– 容器(Composite)构件角色: 有容器特征,可以包含子节点。非叶子节点

组合模式工作流程分析:

– 组合模式为处理树形结构提供了完美的解决方案,描述了如何将容器和叶子进行递归组合,使得用户在使用时可以一致性的对待容器和叶子。

– 当容器对象的指定方法被调用时,将遍历整个树形结构,寻找也包含这个方法的成员,并调用执行。其中,使用了递归调用的机制对整个结构进行处理。

 

开发中的应用场景:

– 操作系统的资源管理器

– GUI中的容器层次图

– XML文件解析

– OA系统中,组织结构的处理

– Junit单元测试框架:底层设计就是典型的组合模式,TestCase(叶子)、TestUnite(容器)、Test接口(抽象)

 

【类图】

 

 

【概念】

/***
 * 抽象组件,叶子节点和非叶节点的共同点。Component:组成部分、成分、部件
 */
package cn.sxt.composite;

public interface Component {
    void operation();

}

//叶子组件节点
interface Leaf extends Component{
    
}
//容器组件,非叶节点
interface Composite extends Component{
    void add (Component c);//增加节点
    void remove(Component c);//删除节点
    Component getChild(int index);//获得孩子节点
    
}

 

【叶子节点和非叶子节点】

/***
 * 抽象文件(AbstractFile)接口,相当于Component接口(抽象组件)
 */
package cn.sxt.composite;


import java.util.ArrayList;
import java.util.List;

public interface AbstractFile {
    void killVirus();
}

class ImageFile implements AbstractFile{//相当于叶子节点Leaf,单个文本中不能再包含其他文件
    private String name;
    
    public ImageFile(String name) {
        this.name=name;
    }
    @Override
    public void killVirus() {
        System.out.println("查杀图片文件:"+name);        
    }
}

class TextFile implements AbstractFile{
    private String name;
    
    public TextFile(String name) {
        this.name=name;
    }
    @Override
    public void killVirus() {
        System.out.println("查杀文本文件:"+name);        
    }
}

class VideoFile implements AbstractFile{
    private String name;
    
    public VideoFile(String name) {
        this.name=name;
    }
    @Override
    public void killVirus() {
        System.out.println("查杀视频文件:"+name);        
    }
}

class Folder implements AbstractFile{//Folder:文件夹 ,容器组件,非叶子节点
    private String name;
    //定义容器,用来存储子节点
    private List list =new ArrayList();
    
    public Folder(String name) {
        this.name=name;
    }
    
    public void add(AbstractFile file) {
        list.add(file);    
    }
    public void remove(AbstractFile file) {
        list.remove(file);    
    }
    public AbstractFile getChild(int index) {
        return list.get(index);
    }

    //杀毒
    public void killVirus() {//递归:自己调用自己
        System.out.println("对文件夹:"+name+",进行查杀。");
        for (AbstractFile absFile : list) {
            absFile.killVirus();
        }
    }
    
}

 

【客户端】

/**
 * 
 */
package cn.sxt.composite;


public class CLient {
    public static void main(String[] args) {
        AbstractFile f2,f3,f4,f5,f6;
        Folder f1=new Folder("古装剧收藏");
        f2=new ImageFile("SongYi.jpg");
        f3=new TextFile("SheDiao.txt");
        f4=new VideoFile("zhifou.mp4");
        f2.killVirus();
        
        f1.add(f2);
        f1.add(f3);
        f1.add(f4);
        Folder f11=new Folder("胡歌的古装剧");
        f5=new ImageFile("xianjian1.avi");
        f6=new TextFile("Shenhua.mp4");
        f11.add(f5);
        f11.add(f6);
        f1.add(f11);//文件夹套文件夹
        f1.killVirus();//无论是文件f2还是文件夹f1都是继承同一个AbstractFile接口中的killVirus()方法,只调用这个方法即可
        
        
    }

}

 


推荐阅读
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • 本文介绍了在Python3中如何使用选择文件对话框的格式打开和保存图片的方法。通过使用tkinter库中的filedialog模块的asksaveasfilename和askopenfilename函数,可以方便地选择要打开或保存的图片文件,并进行相关操作。具体的代码示例和操作步骤也被提供。 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • CSS3选择器的使用方法详解,提高Web开发效率和精准度
    本文详细介绍了CSS3新增的选择器方法,包括属性选择器的使用。通过CSS3选择器,可以提高Web开发的效率和精准度,使得查找元素更加方便和快捷。同时,本文还对属性选择器的各种用法进行了详细解释,并给出了相应的代码示例。通过学习本文,读者可以更好地掌握CSS3选择器的使用方法,提升自己的Web开发能力。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • 本文介绍了C++中省略号类型和参数个数不确定函数参数的使用方法,并提供了一个范例。通过宏定义的方式,可以方便地处理不定参数的情况。文章中给出了具体的代码实现,并对代码进行了解释和说明。这对于需要处理不定参数的情况的程序员来说,是一个很有用的参考资料。 ... [详细]
  • 本文介绍了Perl的测试框架Test::Base,它是一个数据驱动的测试框架,可以自动进行单元测试,省去手工编写测试程序的麻烦。与Test::More完全兼容,使用方法简单。以plural函数为例,展示了Test::Base的使用方法。 ... [详细]
  • 关键词:Golang, Cookie, 跟踪位置, net/http/cookiejar, package main, golang.org/x/net/publicsuffix, io/ioutil, log, net/http, net/http/cookiejar ... [详细]
  • 本文介绍了如何使用python从列表中删除所有的零,并将结果以列表形式输出,同时提供了示例格式。 ... [详细]
  • [大整数乘法] java代码实现
    本文介绍了使用java代码实现大整数乘法的过程,同时也涉及到大整数加法和大整数减法的计算方法。通过分治算法来提高计算效率,并对算法的时间复杂度进行了研究。详细代码实现请参考文章链接。 ... [详细]
  • 本文介绍了Android 7的学习笔记总结,包括最新的移动架构视频、大厂安卓面试真题和项目实战源码讲义。同时还分享了开源的完整内容,并提醒读者在使用FileProvider适配时要注意不同模块的AndroidManfiest.xml中配置的xml文件名必须不同,否则会出现问题。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
author-avatar
枇杷语1314
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有