《设计模式》3.结构型模式


点击进入我的博客

3.1 适配器模式

适配器模式把一个类的接口变换成客户端所期待的另一种接口,使得原本因接口不匹配而无法在一起工作的两个类能够在一起工作。

3.1.1 类的适配器结构

类的适配器

  • 目标(Target)角色:这就是所期待得到的接口,由于是类适配器模式,因此目标不可以是类。
  • 源(Adaptee)角色:现有需要适配的接口。
  • 适配器(Adapter)角色:适配器类是本模式的核心,必须是具体类。
interface Target {void operation1();void operation2();
}class Adaptee {public void operation1() {}
}class Adapter extends Adaptee implements Target {@Overridepublic void operation2() {}
}
类的适配器
  • 类的适配器模式把被的类的API转换成目标类的API。
  • 是通过继承实现的。
类的适配器效果
  • 使用一个具体类把源适配到目标中。这样如果源以及源的子类都使用此类适配,就行不通了。
  • 由于适配器是源的子类,因此可以在适配器中重写源的一些方法。
  • 由于引进了一个适配器类,因此只有一个线路到达目标类,是问题得到简化。

3.1.2 对象的适配器结构

对象适配器

  • 目标(Target)角色:这就是所期待得到的接口,因此目标可以是具体或抽象的类
  • 源(Adaptee)角色:现有需要适配的接口。
  • 适配器(Adapter)角色:适配器类是本模式的核心,必须是具体类。
interface Target {void operation1();void operation2();
}class Adaptee {public void operation1() {}
}class Adapter implements Target {private Adaptee adaptee;public Adapter(Adaptee adaptee) {this.adaptee = adaptee;}@Overridepublic void operation1() {adaptee.operation1();}@Overridepublic void operation2() {// do something}
}
对象的适配器
  • 对象的适配器是通过依赖实现的
  • 推荐使用该方法
对象的适配器效果
  • 一个适配器可以把多种不同的源适配到同一个目标,即同一个适配器可以把源类和它的子类都适配到目标接口
  • 与类的适配器模式相比,要想置换源类的方法就不容易。
  • 增加新的方法方便的多

3.1.3 细节

使用场景
  1. 系统需要使用现有的类,而此类的接口不符合系统的需要
  2. 想要建立一个可以重复使用的类,用于与一些彼此间没有太大关联的一些类,包括一些可能在将来引进的类一起工作。
  3. 对对象的适配器儿模式而言,在设计里,需要改变多个已有的子类的接口,如果使用类的适配器模式,就需要针对每个子类做一个适配器类,这不太实际。
优点
  1. 可以让任何两个没有关联的类一起运行。
  2. 提高了类的复用。
  3. 增加了类的透明度。
  4. 灵活性好。
缺点
  1. 过多地使用适配器,会让系统非常零乱,不易整体进行把握。比如,明明看到调用的是 A 接口,其实内部被适配成了 B 接口的实现,一个系统如果太多出现这种情况,无异于一场灾难。因此如果不是很有必要,可以不使用适配器,而是直接对系统进行重构。
  2. 由于 JAVA 至多继承一个类,所以至多只能适配一个适配者类,而且目标类必须是抽象类。
注意点
  1. 目标接口可以忽略,此时目标接口和源接口实际上是相同的
  2. 适配器类可以是抽象类
  3. 可以有带参数的适配器模式

3.1.4 一个充电器案例

// 充电器只能接受USB接口
public class Charger {public static void main(String[] args) throws Exception{USB usb = new SuperAdapter(new TypeC());connect(usb);usb = new SuperAdapter(new Lightning());connect(usb);}public static void connect(USB usb) {usb.power();usb.data();}
}// 充电器的接口都是USB的,假设有两个方法分别是电源和数据
interface USB {void power();void data();
}// IOS的Lightning接口
class Lightning {void iosPower() {System.out.println("IOS Power");}void iosData() {System.out.println("IOS Data");}
}// TYPE-C接口
class TypeC {void typeCPower() {System.out.println("TypeC Power");}void typeCData() {System.out.println("TypeC Data");}
}// 超级适配器,可以适配多种手机机型
class SuperAdapter implements USB {private Object obj;public SuperAdapter(Object obj) {this.obj = obj;}@Overridepublic void power() {if(obj.getClass() == Lightning.class) {((Lightning)obj).iosPower();} else if(obj.getClass() == TypeC.class) {((TypeC)obj).typeCPower();}}@Overridepublic void data() {if(obj.getClass() == Lightning.class) {((Lightning)obj).iosData();} else if(obj.getClass() == TypeC.class) {((TypeC)obj).typeCData();}}
}

3.2 缺省适配模式

缺省适配模式为一个接口提供缺省实现,这样子类型可以从这个缺省实现进行扩展,而不必从原有接口进行扩展。

3.2.1 缺省适配模式结构

缺省适配模式

简单的例子
  • 下面程序中,Monk接口定义了两个方法,于是它的子类必须实现这两个方法。
  • 但出现了一个LuZhiShen,他只能实现一部分方法,另一部分方法无法实现
  • 所以需要一个抽象的适配类MonkAdapter实现此Monk接口,此抽象类给接口所有方法都提供一个空的方法,LuZhiShen只需要继承该适配类即可。
// 和尚
interface Monk {void practiceKungfu();void chantPrayer();
}abstract class MonkAdapter implements Monk {@Overridepublic void practiceKungfu() {}@Overridepublic void chantPrayer() {}
}class LuZhiShen extends MonkAdapter {@Overridepublic void practiceKungfu() {System.out.println("拳打镇关西");}
}

3.2.2 细节

使用场景
  • 任何时候不准备实现一个接口中所有方法的时候
作用

缺省适配器模式可以使所需要的类不必实现不需要的接口。

核心点
  • 缺省适配的类必须是抽象类,因为这个类不应当被实例化
  • 缺省适配的类提供的方法必须是具体的方法,而不是抽象的方法。

3.3 组合模式

组合模式,就是在一个对象中包含其他对象,这些被包含的对象可能是终点对象(不再包含别的对象),也有可能是非终点对象(其内部还包含其他对象)。
我们将对象称为节点,即一个根节点包含许多子节点,这些子节点有的不再包含子节点,而有的仍然包含子节点,以此类推。很明显,这是树形结构,终结点叫叶子节点,非终节点叫树枝节点,第一个节点叫根节点。

3.3.1 安全式的合成模式结构

安全式的合成模式要求管理集合的方法只出现在树枝结点(Composite)中,而不出现在树叶结点中。
安全式的合成模式

  • 抽象构建(Component)角色:这是一个抽象角色,他给参加组合的对象定义出公共的接口及其默认行为,可以用来管理所有的子对象。
  • 树叶(Leaf)角色:树叶是没有子对象的对象,定义出参加组合的原始对象的行为。
  • 树枝(Composite)角色:代表参加组合的有下级子对象的对象。树枝构件类给出所有管理子对象的方法。

3.3.2 透明的合成模式结构

透明的合成模式
透明的合成模式要求所有的具体构建类,都符合一个固定的接口。

3.4 装饰器模式

装饰器模式(Decorator)允许向一个现有的对象添加新的功能,同时又不改变其结构。这种类型的设计模式属于结构型模式,它是作为现有的类的一个包装。

3.4.1 装饰器结构

装饰器模式

  • 抽象构件(Component)角色:给出一个抽象结构,以规范准备接受附加责任的对象。
  • 具体构件(Concrete Component)角色:定义一个要接受附加责任的类
  • 装饰(Decorator)角色:持有一个构件对象的实例,并定义一个与抽象构件接口一致的接口。
  • 具体装饰(Concrete Decorator)角色:负责给构件对象“贴上”附加的责任。

3.4.2 装饰器细节

使用场景
  1. 需要扩展一个类的功能
  2. 需要动态地给一个对象增加功能,这些功能可以再动态的插销
  3. 需要增加由一些基本功能的排列组合而产生非常大量的功能
优点
  1. 更加灵活:装饰模式和继承关系的目的都是要扩展对象的功能,但是装饰模式比继承更加灵活
  2. 多样性:通过使用不同具体装饰类及其排列组合,可以创造出不同的行为
  3. 动态扩展:装饰器可以动态扩展构件类
缺点
  1. 会产生比继承关系更多的对象
  2. 比继承更加容易出错
注意点
  1. 装饰类的接口必须与被装饰类的接口相容。
  2. 尽量保持抽象构件(Component)简单。
  3. 可以没有抽象的(Component),此时装饰器(Decorator)一般是具体构件(Concrete Component)的一个子类。
  4. 装饰(Decorator)和具体装饰(Concrete Decorator)可以合并。
InputStream及其子类

InputStream

  • 抽象构件(Component)角色:InputStream
  • 具体构件(Concrete Component)角色:ByteArrayInputStreamPipedInputStreamStringBufferInputStream等原始流处理器。
  • 装饰(Decorator)角色:FilterInputStream
  • 具体装饰(Concrete Decorator)角色:DateInputStreamBufferedInputStreamLineNumberInputStream
OutputStream及其子类

也用到类装饰器模式

3.4.3 例子

// Component:一个艺人
interface Artist {void show();
}// Concrete Component:一个歌手
class Singer implements Artist {@Overridepublic void show() {System.out.println("Let It Go");}
}// 装饰后的歌手:不仅会唱歌,还会讲笑话和跳舞
class SuperSinger implements Artist {private Artist role;public SuperSinger(Artist role) {this.role = role;}@Overridepublic void show() {System.out.println("Tell Jokes!");role.show();System.out.println("Dance!");}
}

3.5 代理模式

代理模式给某一个对象提供一个代理对象,并由代理对象控制对原对象对引用

3.5.1 代理模式结构

代理模式

  • 抽象主题(Subject)角色:声明了真实主题和代理主题的共同接口
  • 代理主题(Proxy)角色:代理主题角色内部含有对真实主题的引用,从而可以在任何时候操作真实主题对象;代理主题角色提供一个与真实主题角色相同的接口,以便可以在任何时候都可以替代真实主题控制对真实主题的引用,负责在需要的时候创建真实主题对象(和删除真实主题对象);代理角色通常在将客户端调用传递给真实的主题之前或之后,都要执行某个操作,而不是单纯地将调用传递给真实主题对象。
  • 真实主题(RealSubject)角色:定义了代理角色所代表的真实对象。
public class Test {public static void main(String[] args) {Subject subject = new RealSubject();Subject proxy = new ProxySubject(subject);proxy.request(); // 此处通过代理类来执行}
}interface Subject {void request();
}class RealSubject implements Subject {@Overridepublic void request() {System.out.println("RealSubject");}
}class ProxySubject implements Subject {private Subject subject;public ProxySubject(Subject subject) {this.subject = subject;}@Overridepublic void request() {System.out.println("ProxySubject");}
}

3.5.2 动态代理

自从JDK 1.3以后,Java在java.lang.reflect库中提供了一下三个类直接支持代理模式:ProxyInvocationHanderMethod

动态代理步骤
  1. 创建一个真实对象
  2. 创建一个与真实对象有关的调用处理器对象InvocationHandler
  3. 创建代理,把调用处理器和要代理的类联系起来Proxy.newInstance()
  4. 在调用处理对象的invoke()方法中执行相应操作
public class Test {public static void main(String[] args) {// 创建要被代理的实例对象Subject subject = new RealSubject();// 创建一个与被代理实例对象有关的InvocationHandlerInvocationHandler handler = new ProxySubject(subject);// 创建一个代理对象来代理subject,被代理的对象subject的每个方法执行都会调用代理对象proxySubject的invoke方法Subject proxySubject = (Subject) Proxy.newProxyInstance(Subject.class.getClassLoader(), new Class[]{Subject.class}, handler);// 代理对象执行proxySubject.request();}
}interface Subject {void request();
}class RealSubject implements Subject {@Overridepublic void request() {System.out.println("RealSubject");}
}class ProxySubject implements InvocationHandler {private Subject subject;public ProxySubject(Subject subject) {this.subject = subject;}/*** @param proxy 要代理的* @param method* @param args* @return*/@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("Before Proxy");Object obj = method.invoke(subject, args);System.out.println("After Proxy");return obj;}
}
  • 可以使用范型来创建ProxySubject
  • 可以使用匿名内部类减少代码数量请查看14.7节

3.5.3 细节

优点
  1. 代理类和真实类分离,职责清晰。
  2. 在不改变真是累代码的基础上扩展了功能。
缺点
  1. 由于在客户端和真实主题之间增加了代理对象,因此有些类型的代理模式可能会造成请求的处理速度变慢。
  2. 实现代理模式需要额外的工作,有些代理模式的实现非常复杂。
和适配器模式的关系

适配器模式的用意是改变所考虑对象的接口,而代理模式不能改变。

和装饰模式
  • 装饰模式应当为所装饰的对象提供增强功能
  • 代理模式对对象的使用施加控制,并不提供对象本身的增强功能
虚拟代理
  • 虚拟代理模式(Virtual PRoxy)会推迟真正所需对象实例化时间。在需要真正的对象工作之前,如果代理对象能够处理,那么暂时不需要真正对象来出手。
  • 当一个真实主题对象的加载需要耗费资源时,一个虚拟代理对象可以代替真实对象接受请求,并展示“正在加载”的信息,并在适当的时候加载真实主题对象。

3.6 享元模式

享元模式以共享的方式高效地支持大量的细粒度对象。

3.6.1 单纯享元模式

单纯享元模式中,所有的享元对象都是可以共享的。
单纯享元模式

  • 抽象享元(Flyweight)角色:是所有具体享元角色的超类,并为这些类规定公共接口。
  • 具体享元(Concrete Flyweight)角色:实现抽象享元的接口。如果由内蕴状态的话,必须负责为内蕴状态提供空间。
  • 享元工厂(Flyweight Factory)角色:负责创建和管理享元角色。如果系统中有了则返回该角色,没有则创建。
  • 客户端(Client)角色:维护一个所有享元对象的引用。存储所有享元对象的外蕴状态。

3.6.2 复合享元模式

复合享元模式

  • 抽象享元(Flyweight)角色 :给出一个抽象接口,以规定出所有具体享元角色需要实现的方法。
  • 具体享元(ConcreteFlyweight)角色:实现抽象享元角色所规定出的接口。如果有内蕴状态的话,必须负责为内蕴状态提供存储空间。
  • 复合享元(ConcreteCompositeFlyweight)角色 :复合享元角色所代表的对象是不可以共享的,但是一个复合享元对象可以分解成为多个本身是单纯享元对象的组合。复合享元角色又称作不可共享的享元对象(UnsharedConcreteFlyweight)。
  • 享元工厂(FlyweightFactory)角色 :负责创建和管理享元角色。当一个客户端对象调用一个享元对象的时候,如果已经有了,享元工厂角色就应当提供这个已有的享元对象;如果系统中没有一个适当的享元对象的话,享元工厂角色就应当创建一个合适的享元对象。

3.6.3 细节

内蕴状态和外蕴状态

内蕴状态:是存储在享元对象内部的,不会随环境改变而改变的。一个享元可以具有内蕴状态并可以共享。
外蕴状态:随环境改变而改变、不可以共享的状态。享元对象的外蕴状态必须由客户端保存,并在享元对象被创建之后,在需要使用的时候再传入到享元对象内部。

不变模式

享元模式中的对象不一定非要是不变对象,但大多数享元对象的确是这么设计的。

享元工厂
  1. 使用单例模式:一般只需要一个享元工厂,可以设计成单例的。
  2. 备忘录模式:享元工厂负责维护一个表,通过这个表把很多相同的实例与它们的一个对象联系起来。
优点

减少对象的创建,降低内存消耗

缺点
  1. 提高了系统的复杂度,为了使对象可以共享,需要将一些状态外部化
  2. 需要将一些状态外部化,而读取外部状态是的运行时间稍微变长
使用场景
  1. 一个系统中有大量对象。
  2. 这些对象消耗大量内存。
  3. 这些对象的状态大部分可以外部化。
  4. 这些对象可以按照内蕴状态分为很多组,当把外蕴对象从对象中剔除出来时,每一组对象都可以用一个对象来代替。
  5. 系统不依赖于这些对象身份,换言之,这些对象是不可分辨的。
Java应用
  1. String对象,有则返回,没有则创建一个字符串并保存
  2. 数据库的连接池

3.6.4 案例

依旧是熟悉的KFC点餐为例:

  • 外蕴状态:点餐的顾客
  • 内蕴状态:顾客点的食物
  • 具体享元角色:维护内蕴状态(客人要点的食物)。
public class KFC {public static void main(String[] args) {OrderFactory orderFactory = OrderFactory.getInstance();Order order = orderFactory.getOrder(Food.MiniBurger);order.operation("李雷");order = orderFactory.getOrder(Food.MiniBurger);order.operation("韩梅梅");}
}enum Food {MiniBurger,MexicanTwister,CornSalad,HotWing,PepsiCola
}// Flyweight角色
interface Order {// 传入的是外蕴对象:顾客void operation(String customer);
}// ConcreteFlyweight角色
class FoodOrder implements Order {// 内蕴状态private Food food;// 构造方法,传入享元对象的内部状态的数据public FoodOrder(Food food) {this.food = food;}@Overridepublic void operation(String customer) {System.out.println("顾客[" + customer + "]点的是" + food.toString());}
}// FlyweightFactory角色
class OrderFactory {private Map<Food, Order> orderPool = new HashMap<>();private static OrderFactory instance = new OrderFactory();private OrderFactory() {}public static OrderFactory getInstance() {return instance;}// 获取Food对应的享元对象public Order getOrder(Food food) {Order order = orderPool.get(food);if (null == order) {order = new FoodOrder(food);orderPool.put(food, order);}return order;}
}

3.7 门面模式

门面模式(Facade Pattern)要求一个子系统的外部与其内部通信,必须通过一个统一的门面对象进行。

3.7.1 门面模式结构

门面模式没有一个一般化的类图描述,可以用下面的例子来说明。
门面模式

  • 门面(Facade)角色:外部可以调用这个角色的方法。此角色知道子系统的功能和责任。
  • 子系统(Subsystem)角色:可以有多个子系统,子系统不需要知道门面的存在。

3.7.2 细节

门面数量

通常只需要一个门面类,而且只有一个实例,因此可以设计称单例模式。当然也可有多个类。

使用场景
  1. 为一个复杂的子系统提供一个简单的接口
  2. 使子系统和外部分离开来
  3. 构建一个层次化系统时,可以使使用Facade模式定义系统中每一层,实现分层。
优点
  1. 减少系统之间的相互依赖。
  2. 提高了安全性。
缺点
  1. 不符合开闭原则
  2. 如果要改东西很麻烦,继承重写都不合适。
Java例子

MVC三层结构

3.7.3 KFC例子

假如没有服务员(门面),顾客(外部系统)要点一个套餐需要知道每个套餐包含的食物(子系统)种类,这样就会非常麻烦,所以最好的方式是直接告诉服务员套餐名称就好了。

public class Customer {public static void main(String[] args) {Waiter waiter = new Waiter();List<Food> foodList = waiter.orderCombo("Combo1");}
}abstract class Food {}
class MiniBurger extends Food {}
class MexicanTwister extends Food {}
class CornSalad extends Food {}
class HotWing extends Food {}
class PepsiCola extends Food {}class Waiter {public List<Food> orderCombo(String comboName) {List<Food> foodList;switch (comboName) {case "Combo1" : foodList = Arrays.asList(new MiniBurger(), new CornSalad(), new PepsiCola()); break;case "Combo2":foodList = Arrays.asList(new MexicanTwister(), new HotWing(), new PepsiCola());break;default:foodList = new ArrayList<>();}return foodList;}
}

3.8 过滤器模式

过滤器模式使用不同的条件过滤一组对象,并通过逻辑操作以解耦方式将其链接。这种类型的设计模式属于结构模式,因为该模式组合多个标准以获得单个标准。

3.8.1 细节

步骤
  1. 创建一个要过滤的普通类,要有获得其私有属性的get方法
  2. 创建一个接口,规定过滤方法
  3. 实现接口,可以依需要重写过滤方法,参数传递的一般是存储过滤类的容器类
  4. 复杂过滤类可以通过设置传递接口参数(复用其他基础过滤类)来实现多重过滤
Java8

Java8中的lambda表达式可以更简单的实现过滤器

List<Movie> movies = Stream.of(new Movie("大话西游","comedy"),new Movie("泰囧", "comedy"),new Movie("禁闭岛", "suspense")).filter(var -> "comedy".equals(var.getType())).collect(Collectors.toList());

3.8.2 电影的例子

  1. 创建被过滤的类Movie,根据它的type属性实现过滤
  2. 创建接口Criteria,规定过滤方法
  3. 创建喜剧电影过滤器ComedyMovieCriteria,根据comedy==movie.type来过滤出需要的喜剧电影
public class Test {public static void main(String[] args) {List<Movie> movies = new ArrayList(){{add(new Movie("大话西游","comedy"));add(new Movie("泰囧", "comedy"));add(new Movie("禁闭岛", "suspense"));}};System.out.println(new ComedyMovieCriteria().meetCriteria(movies));}
}// 被筛选的对象
class Movie {private String name;// 电影类型private String type;public Movie(String name, String type) {this.name = name;this.type = type;}// getters & setters & toString
}// 过滤器接口
interface Criteria {/*** @param movies 要被筛选的电影* @return 筛选后的结果*/List<Movie> meetCriteria(List<Movie> movies);
}// 过滤喜剧电影的过滤器,要求是movie.type==comedy
class ComedyMovieCriteria implements Criteria {@Overridepublic List<Movie> meetCriteria(List<Movie> movies) {List<Movie> result = new ArrayList<>();for (Movie movie : movies) {if ("comedy".equals(movie.getType())) {result.add(movie);}}return result;}
}

3.9 桥接模式

桥接模式是将抽象化实现化解耦,使两者可以独立地变化。桥接模式有助于理解面向对象的设计原则,包括开闭原则以及组合聚合复用原则

3.9.1 桥接模式结构

桥接模式

这个系统含有两个等级结构
  • 由抽象化角色和修正抽象化角色组成的抽象化等级结构。
  • 由实现化角色和两个具体实现化角色所组成的实现化等级结构。
桥接模式所涉及的角色
  • 抽象化(Abstraction)角色:抽象化给出的定义,并保存一个对实现化对象的引用。
  • 修正抽象化(Refined Abstraction)角色:扩展抽象化角色,改变和修正父类对抽象化的定义。
  • 实现化(Implementor)角色:这个角色给出实现化角色的接口,但不给出具体的实现。必须指出的是,这个接口不一定和抽象化角色的接口定义相同,实际上,这两个接口可以非常不一样。实现化角色应该只给出底层操作,而抽象化角色应该只给出基于底层操作的更高一层的操作。
  • 具体实现化(Concrete Implementor)角色:这个角色给出实现化角色接口的具体实现。

3.9.2 细节

抽象化、实现化、解耦

抽象化:存在于多个实体中的共同的概念性联系;通过忽略一些信息,把不同的实体当作相同的实体来对待。
实现化:抽象化给出的具体实现就是实现化。一个类的实例就是这个类的实现化,一个子类就是它超类的实现化。
解耦:耦合就是两个实体的某种强关联,把它们的强关联去掉就是解耦。
强关联与弱关联:所谓强关联,就是在编译期已经确定的,无法在运行期动态改变的关联;所谓弱关联,就是可以动态地确定并且可以在运行期动态地改变的关联。继承是强关联,而聚合关系是弱关联

核心理解

桥接模式中的脱耦,就是在抽象化和实现化之间使用组合关系而不是继承关系,从而使两者可以相对独立的变化。

优点
  1. 实现抽象化和实现化的分离。
  2. 提高了代码的扩展能力。
  3. 实现细节对客户透明。
缺点
  1. 桥接模式的引入会增加系统的理解与设计难度
  2. 由于聚合关联关系建立在抽象层,要求开发者针对抽象进行设计与编程。
使用场景
  1. 如果一个系统需要在构件的抽象化角色和具体化角色之间增加更多的灵活性,避免在两个层次之间建立静态的继承联系,通过桥接模式可以使它们在抽象层建立一个关联关系。
  2. 抽象化角色和实现化角色可以以继承的方式独立扩展而互不影响,在程序运行时可以动态将一个抽象化子类的对象和一个实现化子类的对象进行组合,即系统需要对抽象化角色和实现化角色进行动态耦合。
  3. 一个类存在两个独立变化的维度,且这两个维度都需要进行扩展。
  4. 虽然在系统中使用继承是没有问题的,但是由于抽象化角色和具体化角色需要独立变化,设计要求需要独立管理这两者。
  5. 对于那些不希望使用继承或因为多层次继承导致系统类的个数急剧增加的系统,桥接模式尤为适用
Java例子

大多数的驱动器(Driver)都是桥接模式的应用,使用驱动程序的应用系统就是抽象化角色,而驱动器本身扮演实现化角色。
JDBC驱动器

3.9.3 发送消息的案例

  • 下面案例中,SendMsg及其子类是按照发送消息的方式进行扩展的;而Send是按照发送消息的时间进行扩展的,两者互不影响。
  • Send持有类一个SendMsg对象,并可以使用此对象的方法。
// Implementor角色
interface SendMsg {void sendMsg();
}// Concrete Implementor角色
class EmailSendMsg implements SendMsg {@Overridepublic void sendMsg() {System.out.println("Send Msg By Email");}
}// Concrete Implementor角色
class WeChatSendMsg implements SendMsg {@Overridepublic void sendMsg() {System.out.println("Send Msg By WeChat");}
}// Abstraction 角色
abstract class Send {protected SendMsg sendMsg;public Send(SendMsg sendMsg) {this.sendMsg = sendMsg;}public abstract void send();
}// Concrete Implementor角色
class ImmediatelySend extends Send {public ImmediatelySend(SendMsg sendMsg) {super(sendMsg);}@Overridepublic void send() {sendMsg.sendMsg();System.out.println("Send Msg Immediately");}
}// Concrete Implementor角色
class DelayedlySend extends Send {public DelayedlySend(SendMsg sendMsg) {super(sendMsg);}@Overridepublic void send() {sendMsg.sendMsg();System.out.println("Send Msg DelayedlySend");}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/281447.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

最快的计算机操作,世界十大最快的超级计算机

最近&#xff0c;《联邦储备技术》杂志对全球超级计算机进行了排名&#xff0c;并从中选出了十个最快的超级计算机. 其中&#xff0c;中国有两台超级计算机进入了榜单&#xff0c;而“天河2号”则依靠双精度浮点算术峰. 速度达到了每秒5490亿次&#xff0c;占据了王位.这也是两…

苹果iOS 10.3.1修复博通Wi-Fi芯片重大安全漏洞

如果你还没有将设备升级到 iOS 10.3.1 的话&#xff0c;那么现在是个机会了。因为不久前发布的 iOS 10.3.1&#xff0c;修复了 iPhone 中博通 Wi-Fi 芯片的一个重大安全漏洞&#xff0c;该安全漏洞可能会使在 Wi-Fi 范围内的攻击者在智能手机上注入并运行代码。 Google Project…

台积电放大招:甩开英特尔 7nm和5nm芯片将诞生

北京时间1月20日消息&#xff0c;据科技网站AppleInsider报道&#xff0c;近几年来台积电的发展势头相当猛&#xff0c;该公司总裁兼联合CEO刘德音(Mark Liu)在最近的投资者会议中表示&#xff0c;预计今年年末公司就将正式量产10nm晶圆。此外&#xff0c;台积电7nm研发一如预期…

01: 实现注册登录功能

目录&#xff1a;抽屉项目之js最佳实践 01: 实现注册登录功能 02: 实现发布帖子功能 03: 将帖子展示到页面、点赞 04: 层级评论 目录&#xff1a; 1.1 显示、隐藏 "登录/注册" 菜单1.2 注册功能1.3 登录功能1.4 获取当前用户数量1.1 显示、隐藏 "登录/注册"…

计算机系统怎么算页面大小,电脑网页的设计尺寸是多少

电脑网页的设计尺寸是多少刚入门的网页设计师可能对电脑网页的设计尺寸大小存在疑问&#xff0c;以下百分网小编整理的电脑网页的设计尺寸&#xff0c;希欢迎阅读!  对大于30W台客户端用户进行测试&#xff0c;得到的测试数据如下(数据来源于网络)&#xff1a;安全分辨率为10…

WPF 托盘闪烁

WPF 托盘闪烁控件名&#xff1a;NotifyIcon作者&#xff1a;WPFDevelopersOrg - 弈虎、驚鏵原文链接&#xff1a; https://github.com/WPFDevelopersOrg/WPFDevelopers框架使用大于等于.NET40。Visual Studio 2022。项目使用 MIT 开源许可协议。接着上一篇基础托盘。新增如下…

Information Retrieval 倒排索引 学习笔记

一&#xff0c;问题描述 在Shakespeare文集&#xff08;有很多文档Document&#xff09;中&#xff0c;寻找哪个文档包含了单词“Brutus”和"Caesar"&#xff0c;且不包含"Calpurnia"。这其实是一个查询操作&#xff08;Boolean Queries&#xff09;。 在U…

计算机地址栏搜索不了网,我的电脑地址栏不见了怎么办 地址栏不见了如何解决...

导语&#xff1a;小编对电脑是比较痴迷的&#xff0c;因此喜欢在自己的电脑上进行各种操作&#xff0c;也经常会碰到一些问题。今天要为大家介绍的是在我的电脑地址栏不见了之后怎么办&#xff0c;熟悉电脑的朋友都能够了解。在我的电脑主界面里面&#xff0c;有一个地址栏&…

实践App内存优化:如何有序地做内存分析与优化

由于项目里之前线上版本出现过一定比例的OOM,虽然比例并不大&#xff0c;但是还是暴露了一定的问题&#xff0c;所以打算对我们App分为几个步骤进行内存分析和优化&#xff0c;当然内存的优化是个长期的过程&#xff0c;不是一两个版本的事&#xff0c;每个版本都需要收集线上内…

WinForm(十四)窗体滚动日志

在桌面程序里&#xff0c;一般日志记录到文件里就可以了&#xff0c;但有的时间&#xff0c;也需要在窗体上动态滚动显示&#xff0c;这时&#xff0c;就需要引入日志框架了。这里引入的依旧是NLog&#xff08;在我的Mini API系统里&#xff0c;用的也是NLog&#xff09;。首先…

xp计算机找不到音量调节,WinXP电脑没声音且小喇叭不见了如何解决?

有用户在使用电脑听音乐的时候&#xff0c;突然发现电脑没有声音了&#xff0c;本来以为只是被禁了音&#xff0c;想着调节音量即可解决问题。但是当他想要点开音量小喇叭的时候&#xff0c;发现桌面任务栏通知区域的小喇叭不见了&#xff0c;这该怎么办呢&#xff1f;下面小编…

2018-2019-1 20165211 实验四 外设驱动程序设计

2018-2019-1 20165211 实验四 外设驱动程序设计 任务一 1.实验要求 学习资源中全课中的“hqyj.嵌入式Linux应用程序开发标准教程.pdf”中的第十一章 提交康奈尔笔记的照片&#xff08;可以多张&#xff09; 2. 任务完成 任务二 1. 实验要求 在Ubuntu完成资源中全课中的“hqyj.嵌…

《ASP.NET Core 6框架揭秘》实例演示[31]:路由高阶用法

ASP.NET的路由是通过EndpointRoutingMiddleware和EndpointMiddleware这两个中间件协作完成的&#xff0c;它们在ASP.NET平台上具有举足轻重的地位&#xff0c;MVC和gRPC框架&#xff0c;Dapr的Actor和发布订阅编程模式都建立在路由系统之上。Minimal API更是将提升到了前所未有…

java中文乱码解决之道(五)—–java是如何编码解码的

编码&解码 1&#xff1a;I/O操作 2&#xff1a;内存 3&#xff1a;数据库 4&#xff1a;javaWeb 下面主要介绍前面两种场景&#xff0c;数据库部分只要设置正确编码格式就不会有什么问题&#xff0c;javaWeb场景过多需要了解URL、get、POST的编码&#xff0c;servlet的解码…

win10系统按esc会弹出计算机,win10系统版本2004控制面板多出ESC是什么原因?

如果我们的电脑在升级了win102004控制面板多出ESC什么情况方法一&#xff1a;“干净启动”&#xff0c;排除第三方软体的影响1.停止非核心的程序运作(包括第三方杀毒、优化软体)2.情况允许的话&#xff0c;卸载设备中的第三方杀毒、管家、优化软件3.同时按【4.点击【服务】>…

CentOS6/7 配置守护进程

CentOS6.xCentOS6中转用Upstrat代替以前的init.d/rcX.d的线性启动方式。一、相关命令通过initctl help可以查看相关命令[rootlocalhost ~]# initctl help Job commands:start Start job.stop Stop job.restart …

Java并发(二十一):线程池实现原理

一、总览 线程池类ThreadPoolExecutor的相关类需要先了解&#xff1a; &#xff08;图片来自&#xff1a;https://javadoop.com/post/java-thread-pool#%E6%80%BB%E8%A7%88&#xff09; Executor&#xff1a;位于最顶层&#xff0c;只有一个 execute(Runnable runnable) 方法&a…

进程池

转自&#xff1a;https://www.cnblogs.com/kaituorensheng/p/4465768.html 在利用Python进行系统管理的时候&#xff0c;特别是同时操作多个文件目录&#xff0c;或者远程控制多台主机&#xff0c;并行操作可以节约大量的时间。当被操作对象数目不大时&#xff0c;可以直接利用…

gulp版本号管理插件注意事项

2019独角兽企业重金招聘Python工程师标准>>> 打开node_modules\gulp-rev\index.js 第144行 manifest[originalFile] revisionedFile; 更新为: manifest[originalFile] originalFile ?v file.revHash; 打开node_modules\rev-path\index.js 第10行 return filena…

bigfile.to服务器位置,Cloudera Manager 迁移服务器

Cloudera Manager还是比较耗资源的&#xff0c;想把Cloudera Manager&#xff0c;移动到比较好的机器上。在这篇文章中&#xff0c;Cloudera Manager安装在bigserver1上面&#xff0c;bigserver1是奔腾双核的CPU。1&#xff0c;Cloudera Manager占资源比较多cloudera manager占…