欢迎访问shiker.tech

请允许在我们的网站上展示广告

您似乎使用了广告拦截器,请关闭广告拦截器。我们的网站依靠广告获取资金。

结构型设计模式实战【一】-适配器, 桥接与组合
(last modified Dec 28, 2024, 12:05 AM )
by
侧边栏壁纸
  • 累计撰写 194 篇文章
  • 累计创建 66 个标签
  • 累计收到 4 条评论

目 录CONTENT

文章目录

结构型设计模式实战【一】-适配器, 桥接与组合

橙序员
2022-08-20 / 0 评论 / 0 点赞 / 600 阅读 / 3,503 字 / 正在检测百度是否收录... 正在检测必应是否收录...
文章摘要(AI生成)

结构型设计模式包括适配器模式、桥接模式、组合模式、外观模式、享元模式、代理模式、装饰器模式。适配器模式用于转换类的接口,使得原本不兼容的接口可以协同工作。适配器模式的优点在于增加类的透明性、提高复用度和灵活性。适配器模式适用于修改已投产接口、系统扩展使用新类等情况。桥接模式将类的抽象部分和实现分离,实现了抽象和实现的解耦及扩展能力。桥接模式适用于不适用继承、接口或抽象类不稳定的场景。总体而言,结构型设计模式可以有效提高系统的灵活性和扩展性,在实际应用中能解决接口不兼容、需求变更等问题。

结构型设计模式概述

结构型设计模式用于处理类或对象的组合,对类的拓展和封装提供指南,包括适配器模式、桥接模式、组合模式、外观模式、享元模式、代理模式、装饰器模式

适配器模式

定义

将一个类的接口转换成希望的另外一个接口,使得原本不兼容的接口可以协同工作

角色

  • 现有实例(adaptee):被适配的角色,也指业务已经存在的实际情况
  • 目标接口(target):定义所需的方法,即新场景下需要实现的场景
  • 适配器(adapter):基于现有实例,转换成所需的目标接口
  • 请求者(client):负责使用目标接口进行具体处理

image-20220820162639140

优点

  • 适配器模式可以让两个没有任何关系的类在一起运行,只要适配器这个角色能够搞定 他们就成。
  • 增加了类的透明性 想想看,我们访问的Target目标角色,但是具体的实现都委托给了源角色,而这些对高 层次模块是透明的,也是它不需要关心的。
  • 提高了类的复用度 当然了,源角色在原有的系统中还是可以正常使用,而在目标角色中也可以充当新的演员。
  • 灵活性非常好 某一天,突然不想要适配器,没问题,删除掉这个适配器就可以了,其他的代码都不用修改,基本上就类似一个灵活的构件,想用就用,不想就卸载。

使用场景

适配器应用的场景只要记住一点就足够了:你有动机修改一个已经投产中的接口时,适 配器模式可能是最适合你的模式。比如系统扩展了,需要使用一个已有或新建立的类,但这 个类又不符合系统的接口,怎么办?使用适配器模式,这也是我们例子中提到的。

最佳实践

适配器模式是一个补偿模式,或者说是一个“补救”模式,通常用来解决接口不相容的问 题,在百分之百的完美设计中是不可能使用到的,什么是百分之百的完美设计?“千虑”而没 有“一失”的设计,但是,再完美的设计也会遇到“需求”变更这个无法逃避的问题,就以我们 上面的人力资源管理系统为例来说,不管系统设计得多么完美,都无法逃避新业务的发生, 技术只是一个工具而已,是因为它推动了其他行业的进步和发展而具有了价值,通俗地说, 技术是为业务服务的,因此业务在日新月异变化的同时,也对技术提出了同样的要求,在这 种要求下,就需要我们有一种或一些这样的补救模式诞生,使用这些补救模式可以保证我们 的系统在生命周期内能够稳定、可靠、健壮的运行,而适配器模式就是这样的一个“救世主”,它在需求巨变、业务飞速而导致你极度郁闷、烦躁、崩溃的时候横空出世,它通过把 非本系统接口的对象包装成本系统可以接受的对象,从而简化了系统大规模变更风险的存 在

使用spring实现

下面场景是已有广告牌场景有打印功能,借助打印功能实现打印机中的打印功能

现有实例

定义一个banner,用来表示广告牌,对输入的字符进行广告化输入输出

@Component
@Slf4j
public class Banner {

   private String string;

   public void setString(String string) {
      this.string = string;
   }

   public void showWithParen(){
      log.info("("+string+")");
   }
   
   public void showWithAster(){
      log.info("*"+string+"*");
   }
}

目标接口

我们现有场景有弱化字符串显示和强化字符串显示两种场景

public interface Print {

	void setString(String string);
	
	void printWeak();
	
	void printStrong();
}

适配器

我们通过编写适配器,通过继承或委托现有实例的方式完成目标接口的实现

继承方式

@Component
public class PrintBanner extends Banner implements Print{

   @Override
   public void setString(String string) {
      super.setString(string);
   }

   @Override
   public void printWeak() {
      // TODO Auto-generated method stub
      showWithParen();
   }

   @Override
   public void printStrong() {
      // TODO Auto-generated method stub
      showWithAster();
   }

}

委托方式

@Component
public class PrintBanner implements Print{

   //通过创建目标对象获取目标对象的功能
   @Resource
   private Banner banner;

   public void setString(String string){
      banner.setString(string);
   }

   @Override
   public void printWeak() {
      // TODO Auto-generated method stub
      banner.showWithParen();
   }

   @Override
   public void printStrong() {
      // TODO Auto-generated method stub
      banner.showWithAster();
   }
}

场景测试

@Controller
public class AdapterController {

   @Resource
   private Print print;

   @GetMapping("/adapter")
   public String adapter(){
      print.setString("hello");
      print.printWeak();
      print.printStrong();
      return "success";
   }
}

输出结果为:

image-20220820154149315

桥接模式

定义

将类的抽象部分和实现分离,使得他们都可以独立地发生变化。即将类的功能层次结构和实现层次结构连接在一起。

功能层次结构:即在子类中增加新的结构

实现层次结构:即在子类中增加新的实现

优点

  • 抽象和实现分离。这也是桥梁模式的主要特点,它完全是为了解决继承的缺点而提出的设计模式。在该模 式下,实现可以不受抽象的约束,不用再绑定在一个固定的抽象层次上。
  • 优秀的扩充能力。看看我们的例子,想增加实现?没问题!想增加抽象,也没有问题!只要对外暴露的接 口层允许这样的变化,我们已经把变化的可能性减到最小。
  • 实现细节对客户透明。客户不用关心细节的实现,它已经由抽象层通过聚合关系完成了封装。

使用场景

  • 不希望或不适用使用继承的场景

    例如继承层次过渡、无法更细化设计颗粒等场景,需要考虑使用桥梁模式。

  • 接口或抽象类不稳定的场景

    明知道接口不稳定还想通过实现或继承来实现业务需求,那是得不偿失的,也是比较失 败的做法。

  • 重用性要求较高的场景

    设计的颗粒度越细,则被重用的可能性就越大,而采用继承则受父类的限制,不可能出 现太细的颗粒度。

最佳实践

不能说继承不好,它非常好,但是有强侵入的缺点,父类的方法子类一定会有,我们可以扬长避短,对于比较明确不发生变 化的,则通过继承来完成;若不能确定是否会发生变化的,那就认为是会发生变化,则通过 桥梁模式来解决,这才是一个完美的世界。

角色

  • 抽象实例(abstraction):位于类的功能层次结构的最上层,定义了基本的功能
  • 拓展抽象实例(RefinedAbstraction):在抽象实例的基础上增加了新功能的角色。
  • 实现者(Implementor):位于实现层次结构的最上层,定义了用于实现抽象实例的接口方法
  • 具体实现者(ConcreteImplementor):负责实现在实现者中定义的接口

image-20220820162346857

使用spring实现

程序的现状是已有显示接口去打开文件、打印、显示等功能,需要拓展重复打印功能

抽象实例

已有显示接口为:

@Component
public class Display {

   @Resource
   private DisplayImpl impl;

   public void setImpl(String string){
      impl.setStringAndWidth(string);
   }

   public void open() {
      impl.rawOpen();
   }

   public void print() {
      impl.rawPrint();
   }

   public void close() {
      impl.rawClose();
   }

   public final void display() {
      open();
      print();
      close();
   }
}

抽象实例实现者

public abstract class DisplayImpl {

   public abstract void setStringAndWidth(String string);

   public abstract void rawOpen();

   public abstract void rawPrint();

   public abstract void rawClose();
}

具体实例实现者

定义使用字符串显示的类

@Component
@Slf4j
public class StringDisplayImpl extends DisplayImpl {

   private String string;
   private int width;

   public void setStringAndWidth(String string){
      this.string = string;
      this.width = string.getBytes().length;
   }

   @Override
   public void rawOpen() {
      // TODO Auto-generated method stub
      printLine();
   }

   @Override
   public void rawPrint() {
      // TODO Auto-generated method stub
      log.info("|" + string + "|");
   }

   @Override
   public void rawClose() {
      // TODO Auto-generated method stub
      printLine();
   }

   private void printLine() {
      // TODO Auto-generated method stub
      StringBuilder stringBuilder = new StringBuilder();
      stringBuilder.append("+");
      for (int i = 0; i < width; i++) {
         stringBuilder.append("-");
      }
      stringBuilder.append("+");
      log.info(String.valueOf(stringBuilder));
   }

拓展实例

拓展后的重复打印的接口为:

@Component
public class CountDisplay extends Display {

   public void mutiDIsplay(int times){
      open();
      for (int i = 0; i < times; i++) {
         print();
      }
      close();
   }

}

场景测试

@RestController
public class BridgeController {

   @Resource
   Display display;
   @Resource
   CountDisplay countDisplay;

   @GetMapping("/bridge")
   public String bridge(){
      display.setImpl("Hello,China!");
      display.display();
      display.setImpl("Hello,World.");
      display.display();
      countDisplay.setImpl("Hello,Universe.");
      countDisplay.display();
      countDisplay.mutiDIsplay(5);
      return "success";
   }
}

测试结果如下

image-20220820163933873

组合模式

定义

将对象组合成树形结构以表示部分-整体的层次结构,使容器与内容具有一致性,创造出递归结构

优缺点

优点

  • 高层模块调用简单

    一棵树形机构中的所有节点都是Component,局部和整体对调用者来说没有任何区别,也就是说,高层模块不必关心自己处理的是单个对象还是整个组合结构,简化了高层模块的代码。

  • 节点自由增加

    使用了组合模式后,我们可以看看,如果想增加一个树枝节点、树叶节点是不是都很容易,只要找到它的父节点就成,非常容易扩展,符合开闭原则,对以后的维护非常有利。

缺点

部分和整体在嵌套使用时,可能会直接调用实现类。在面向接口编程上是很不恰当的,与依赖倒置原则冲 突,在使用的时候要考虑清楚,它限制了你接口的影响范围。

使用场景

  • 维护和展示部分-整体关系的场景,如树形菜单、文件和文件夹管理。
  • 从一个整体中能够独立出部分模块或功能的场景。

最佳实践

组合模式在项目中到处都有,比如现在的页面结构一般都是上下结构,上面放系统的 Logo,下边分为两部分:左边是导航菜单,右边是展示区,左边的导航菜单一般都是树形的 结构,比较清晰,有非常多的JavaScript源码实现了类似的树形菜单,大家可以到网上搜索 一下。

还有,大家常用的XML结构也是一个树形结构,根节点、元素节点、值元素这些都与我 们的组合模式相匹配

角色

  • 内容实例(Leaf):表示内容的角色,在该角色中不能放入其他对象
  • 容器实例(Composite):表示容器的角色,可以在其中放入内容实例和容器实例
  • 组件实例(Component):使内容实例和容器实例具有一致性的角色
  • 客户端(client):调用组合模式的角色

image-20220820164557212

使用spring实现

实现一个能列出文件和文件夹的预览功能

组件实例

定义抽象实例entry,用来实现file和directory的一致性

public abstract class Entry {

   public abstract String getName();

   public abstract int getSize();

   public Entry add(Entry entry) throws FileTreatmentException {
      throw new FileTreatmentException();
   }

   public void printList() {
      printList("");
   }

   protected abstract void printList(String prefix);

   public String toString() {
      return getName() + "(" + getSize() + ")";
   }
}

容器实例

定义文件夹类

@Component
@Scope(value = BeanDefinition.SCOPE_PROTOTYPE)
@Slf4j
public class Directory extends Entry{
    private String name;
    private ArrayList<Entry> directory = new ArrayList<Entry>();

   public void initDirectory(String name) {
      this.name = name;
   }

   @Override
   public String getName() {
      // TODO Auto-generated method stub
      return name;
   }

   @Override
   public int getSize() {
      // TODO Auto-generated method stub
      int size = 0;
      Iterator<Entry> it = directory.iterator();
      while(it.hasNext()){
         Entry entry = it.next();
         size+=entry.getSize();
      }
      return size;
   }
    public Entry add(Entry entry){
       directory.add(entry);
       return this;
    }
   @Override
   protected void printList(String prefix) {
      // TODO Auto-generated method stub
      log.info(prefix+"/"+this);
      Iterator<Entry> it = directory.iterator();
      while (it.hasNext()) {
         Entry object = (Entry) it.next();
         object.printList(prefix+"/"+name);
      }
   }

}

内容实例

定义文件类

@Component
@Scope(value = BeanDefinition.SCOPE_PROTOTYPE)
@Slf4j
public class File extends Entry {

   private String name;
   private int size;

   public void initEntry(String name, int size){
      this.name = name;
      this.size = size;
   }

   @Override
   public String getName() {
      // TODO Auto-generated method stub
      return name;
   }

   @Override
   public int getSize() {
      // TODO Auto-generated method stub
      return size;
   }

   @Override
   protected void printList(String prefix) {
      // TODO Auto-generated method stub
      log.info(prefix + "/" + this);
   }

}

组件生成器

定义文件和文件夹的生成器,方便我们生成多实例

@Component
public abstract class EntryGenerator {

   @Lookup
   public abstract Directory directory();

   public Directory generateDir(String name){
      Directory directory = directory();
      directory.initDirectory(name);
      return directory;
   }

   @Lookup
   public abstract File file();

   public File generateFile(String name, int size){
      File file = file();
      file.initEntry(name, size);
      return file;
   }
}

场景测试

@RestController
@Slf4j
public class CompositeController {

   @Resource
   private EntryGenerator entryGenerator;

   @GetMapping("/composite")
   public String composite(){
      try {
         log.info("Making root entries...");
         Directory rootdir = entryGenerator.generateDir("root");
         Directory bindir = entryGenerator.generateDir("bin");
         Directory tmpir = entryGenerator.generateDir("tmp");
         Directory usrdir = entryGenerator.generateDir("usr");
         rootdir.add(bindir);
         rootdir.add(tmpir);
         rootdir.add(usrdir);
         bindir.add(entryGenerator.generateFile("vi", 10000));
         bindir.add(entryGenerator.generateFile("latex", 20000));
         rootdir.printList();

         log.info("Making user entries...");
         Directory yuki = entryGenerator.generateDir("yuki");
         Directory hanako = entryGenerator.generateDir("hanako");
         Directory tomura = entryGenerator.generateDir("tomura");
         usrdir.add(yuki);
         usrdir.add(hanako);
         usrdir.add(tomura);
         yuki.add(entryGenerator.generateFile("diary.html", 100));
         yuki.add(entryGenerator.generateFile("Composite.java", 200));
         hanako.add(entryGenerator.generateFile("memo.tex", 300));
         tomura.add(entryGenerator.generateFile("game.doc", 400));
         tomura.add(entryGenerator.generateFile("junk.mail", 500));
         rootdir.printList();
      } catch (FileTreatmentException e) {
         // TODO: handle exception
         e.printStackTrace();
      }
      return "success";
   }
}

测试结果如下:

image-20220820181242579

0

评论区