设为首页 - 加入收藏 ASP站长网(Aspzz.Cn)- 科技、建站、经验、云计算、5G、大数据,站长网!
热搜: 重新 试卷 文件
当前位置: 首页 > 运营中心 > 建站资源 > 优化 > 正文

换一种角度:从架构层面来看设计模式

发布时间:2019-07-22 20:06 所属栏目:21 来源:架构思维
导读:大部分讲解设计模式的书或者文章,都是从代码层面来讲解设计模式,看的时候都懂,但是到真正用的时候,还是理不清、想不明。 本文尝试从架构层面来聊一聊设计模式。通过将使用设计模式的代码和不使用设计模式的代码分别放到架构中,来看看设计模式对架构所

大部分讲解设计模式的书或者文章,都是从代码层面来讲解设计模式,看的时候都懂,但是到真正用的时候,还是理不清、想不明。

本文尝试从架构层面来聊一聊设计模式。通过将使用设计模式的代码和不使用设计模式的代码分别放到架构中,来看看设计模式对架构所产生的影响。

一般模式讲解套路

一般讲解设计模式的套路是:

  • 说明模式的意图
  • 说明模式的适用场景
  • 给出模式的类结构
  • 给出对应的代码示例

以策略模式为例:

意图:定义一系列的算法,把它们一个个封装起来, 并且使它们可相互替换。本模式使得算法可独立于使用它的客户而变化。

适用性:

  • 许多相关的类仅仅是行为有异。「策略」提供了一种用多个行为中的一个行为来配置一个类的方法。
  • 需要使用一个算法的不同变体。例如,你可能会定义一些反映不同的空间/时间权衡的算法。当这些变体实现为一个算法的类层次时,可以使用策略模式。
  • 算法使用客户不应该知道的数据。可使用策略模式以避免暴露复杂的、与算法相关的数据结构。
  • 一个类定义了多种行为, 并且这些行为在这个类的操作中以多个条件语句的形式出现。将相关的条件分支移入它们各自的Strategy类中以代替这些条件语句。

类结构:

从架构层面看设计模式

示例代码:

  1. public class Context { 
  2.  
  3.  //持有一个具体策略的对象 
  4.  private Strategy strategy; 
  5.  
  6.  /** 
  7.  * 构造函数,传入一个具体策略对象 
  8.  * @param strategy 具体策略对象 
  9.  */ 
  10.  public Context(Strategy strategy){ 
  11.  this.strategy = strategy; 
  12.  } 
  13.   
  14.  /** 
  15.  * 策略方法 
  16.  */ 
  17.  public void invoke(){ 
  18.  strategy.doInvoke(); 
  19.  } 
  20.  
  21. public interface Strategy { 
  22.  
  23.  /** 
  24.  * 策略方法 
  25.  */ 
  26.  public void doInvoke(); 
  27.  
  28. public class StrategyA implements Strategy { 
  29.  
  30.  @Override 
  31.  public void doInvoke() { 
  32.  System.out.println("InvokeA"); 
  33.  } 
  34.  
  35. public class StrategyB implements Strategy { 
  36.  @Override 
  37.  public void doInvoke() { 
  38.  System.out.println("InvokeB"); 
  39.  } 

从上面的讲解,你能理解策略模式吗?你是否有如下的一些疑问?

  • 使用策略模式和我直接写if-else具体的优势在哪里?
  • if-else不是挺简单的,为什么要多写这么多的类?
  • 如何将Strategy给设置到Context中?
  • 我该如何判断将哪个实现设置给Context?还是ifelse?!那拆成这么多的类不是脱裤子放屁吗?

将模式放入架构中

产生这些疑问的原因,是我们在孤立的看设计模式,而没有把设计模式放到实际的场景中。

当我们将其放到实际项目中时,我们实际是需要一个客户端来组装和调用这个设计模式的,如下图所示:

从架构层面看设计模式  
  1. public class Client { 
  2.   
  3.  public static void main(String[] args) { 
  4.  Strategy strategy; 
  5.  if("A".equals(args[0])) { 
  6.  strategy = new StrategyA(); 
  7.  } else { 
  8.  strategy = new StrategyB(); 
  9.  } 
  10.  Context context = new Context(strategy); 
  11.  context.invoke(); 
  12.  }  

作为比较,这里也给出直接使用ifelse时的结构和代码:

从架构层面看设计模式  
  1. public class Client { 
  2.  public static void main(String[] args) { 
  3.  Context context = new Context(args[0]); 
  4.  context.invoke(); 
  5.  } 
  6.  
  7. public class Context { 
  8.  public void invoke(String type) { 
  9.  if("A".equals(type)) { 
  10.  System.out.println("InvokeA"); 
  11.  } else if("B".equals(type)) { 
  12.  System.out.println("InvokeB"); 
  13.  } 
  14.  } 

乍看之下,使用ifelse更加的简单明了,不过别急,下面我们来对比一下两种实现方式的区别,来具体看看设计模式所带来的优势。

边界不同

(编辑:ASP站长网)

网友评论
推荐文章
    热点阅读