Proxy-代理模式

代理模式

  • 代理模式: Proxy Pattern. 指为对象提供一种代理,用以控制对这个对象的访问. 是一种结构型模式
    Java架构师Day01-源码分析之常用设计模式_Java开发
  • 代理模式包含三种角色:
    • 抽象角色Subject: 声明真实角色和代理角色共同接口方法.该类可以是接口也可以是抽象类
    • 代理角色ProxySubject: 代理类. 代理对象内部包含对真实对象的引用,同时代理对象提供与真实对象相同的接口,可以代替真实对象. 同时,代理对象可以在执行真实对象操作时,附加其余的操作,相当于对真实对象进行封装
    • 真实角色RealSubject: 被代理类. 进行代理的真实对象,负责执行系统的真正的逻辑业务对象.调用真实对象方法,都要经过代理角色进行代理
  • 一般代理可以理解为代码增强,实际上就是在原代码逻辑前后增加一些处理逻辑,而调用者无感知.代理模式属于结构型模式,分为静态代理和动态代理
  • 代理模式通用写法

静态代理

  • 静态代理: 需要定义接口或者父类,被代理的对象和代理对象需要实现相同的接口或者相同的父类
  • 优点:
    • 可以做到在不修改目标对象的功能前提下,对目标功能进行扩展
  • 缺点:
    • 因为代理对象需要与真实对象实现同样的接口,所以会有很多代理类
    • 如果接口增加方法,真实对象与代理对象都要进行维护
  • 静态代理示例

动态代理

  • 动态代理特点:
    • 代理对象不需要实现接口
    • 动态代理是利用JDKAPI, 动态地在内存中构建代理对象.需要指定创建的代理对象或者目标对象实现的接口类型

Java动态代理

  • JavaJDK动态代理API所在的包 : java.lang.reflect.Proxy
  • JDK实现动态代理只需要使用newProxyInstance() 方法:
/**
 * 获取指定接口的代理类实例,该代理类将方法调用分配给指定的调用处理程序
 * 该方法相当于:
 * 	Proxy.getProxyClass(loader, interfaces)
 *       .getConstructor(new Class[] { InvocationHandler.class })
 *       .newInstance(new Object[] { handler })
 *       .getConstructor(new Object[] { InvocationHandler.class });  
 * 
 * @param loader 指定当前真实对象使用的类加载器,获取加载器的方法是固定的
 * @param interfaces 真实对象实现的接口类型,使用泛型的方式确认类型
 * @param h 事件处理,执行真实对象的方法时,会触发事件处理器的方法,会将当前执行目标对象的方法作为参数传入
 */
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h);
  • 动态代理示例
  • Java动态代理要求被代理的对象必须是一个实现了接口的对象:
    • 原因在于Java动态代理生成代理类时默认继承的Proxy
    • 根据Java的单继承多实现的特性,只能实现对接口对象的代理

CGLIB代理

  • 静态代理模式和动态代理模式要求目标对象是一个实现了接口的对象
    • 目标对象有时候只是一个单独的对象,并没有实现任何接口
    • 这时候可以使用目标对象子类的方式实现代理,即CGLIB代理
  • CGLIB代理: 子类代理. 在内存中构建真实对象类的子类对象实现对真实对象的代理
    • Java动态代理的对象必须实现一个或者多个接口.如果需要代理没有是实现接口的类就可以使用CGLIB代理
    • CGLIB是一个高性能的代码生成包,可以在运行时期扩展java类和实现java接口,经常被应用在Spring AOP
    • CGLIB的底层是通过使用一个小而快的字节码处理框架ASM来转换字节码并生成新的类.不推荐直接使用ASM框架,因为这对JVM内部结构包括class文件的格式和指令集有要求
  • CGLIB代理实现:
    • 需要引入cglib依赖,因为Spring中已经是包含cglib功能,所以引入了spring-core依赖即可
    • 引入cglib依赖后,就可以在内存中动态构建子包
    • 代理的类不能是final修饰的类,否则会报错
    • 真实对象的方法如果是final或者static, 就会被拦截,不会执行真实对象的额外业务方法
  • CGLIB代理示例

代理模式和装饰器模式区别

  • UML类图基本没有区别,都是实现同一个接口,一个类包装另一个类

代理模式

  • 控制访问:
    • 为真实对象提供一种代理用于控制对这个对象的访问
    • 在不改变接口的前提下,控制对象的访问

装饰器模式

  • 新增行为:
    • 动态的新增或者组合对象的行为
    • 在不改变接口的前提下,动态扩展对象的功能
  • 装饰器模式的核心: 动态地将属性,功能和责任附加到对象上
Factory-工厂模式

工厂模式

  • 创建型模式:
    • 对类的实例化过程进行抽象,能够将对象的创建和对象的使用分离开来
      • 为了使得软件的结构更加清晰,外界对于这些对象使用只需要知道共同的接口,而不在意具体实现的细节,这样使得整个系统更加符合单一职责的原则
      • 创建型模式隐藏了类的实例的创建细节,通过隐藏对象创建和组合过程从而使得整个系统相互独立的目的
    • 创建型模式在创建什么,由谁创建,何时创建更加灵活
    • 工厂模式是一个重要的创建型模式,主要功能就是实例化对象
  • 工厂模式: 负责将有共同接口的类实例化
    • 主要解决接口选择问题
    • 在不同的条件下需要创建不同的实例时使用
    • 工厂模式是一种创建型模式,提供了创建对象的最佳方式
    • 使用工厂模式创建对象不会对客户端暴露创建逻辑,并且使用一个共同的接口来指向新创建的对象
    • 工厂模式在子类中实现工厂接口,创建过程在子类中执行
  • 工厂模式的分类:
    • 简单工厂模式Simple Factory
    • 工厂方法模式Factory Method
    • 抽象工厂模式Abstract Factory
  • 工厂模式优点:
    • 可以使得代码结构清晰,有效地封装变化
    • 对调用者屏蔽具体的产品类
    • 降低代码的耦合度
  • 工厂模式的使用场景:
    • 在任何需要生成复杂对象的地方,都可以使用工厂方法模式.只有复杂的对象才适用于工厂方法模式.对于简单的只要通过new就可以完成创建的对象,无需使用工厂模式.如果简单对象使用工厂模式,需要引入一个工厂类,增加系统的复杂度
    • 工厂模式是一种典型的解耦模式,当类之间需要增加依赖关系时,可以使用工厂模式降低系统之间的耦合度
    • 工厂模式是依靠抽象架构的,将实例化的任务交给子类实现,扩展性好.当系统需要较好的扩展性时,可以使用工厂模式,不同的产品使用不同的工厂来实现组装

简单工厂模式

  • 简单工厂模式Simple Factory Pattern:
    • 定义一个类用于负责创建其余类的实例,根据自变量的不同返回不同类的实例,被创建的实例通常都有一个共同的父类
    • 简单工厂模式中用于创建实例的方法时静态static方法,因此又称作是静态工厂方法模式
  • 简单工厂模式的角色:
    • 工厂类Factory : 简单工厂模式核心类. 负责创建所有产品的内部逻辑,工厂类可以被外部调用,创建所需对象
    • 抽象产品类Product : 工厂类创建的所有对象的父类,封装产品的共有方法.提高系统的灵活性.使得工厂类只需要定义一个通用的工厂方法,因为所有创建的具体产品都是这个子类对象
    • 具体产品类ConcorrectProduct: 所有被创建的对象都是这个类的具体实例,需要实现抽象产品中声明的抽象方法
      Java架构师Day01-源码分析之常用设计模式_Java开发_02
  • 简单工厂模式代码实现
  • 简单工厂模式优点:
    • 简单工厂模式提供了专门的类用于创建对象,实现了对责任的分割. 工厂类Factory中含有必要的判断逻辑,决定创建具体产品类ConcreteProduct的实例,客户端只需要消费产品
    • 客户端不需要知道需要创建的具体产品类ConcreteProduct的类名,只需要知道具体产品类ConcreteProduct对应的参数即可
    • 通过引入配置文件,可以在不修改客户端的情况下修改和增加新的产品类ConcreteProduct, 提高了系统的灵活性
  • 简单工厂模式缺点:
    • 工厂类Factory集中了所有产品的创建逻辑,如果发生异常,整个系统都会发生故障
    • 简单工厂模式中增加了系统中类的个数,增加了系统的复杂度和理解难度
    • 简单工厂模式中如果需要添加新的产品需要修改工厂逻辑,违背了开闭原则,不利于系统的扩展和维护
    • 简单工厂模式使用了静态方法,无法形成基于继承的等级结构
  • 简单工厂模式的使用场景:
    • 工厂类中负责创建的对象比较少时
    • 客户端只需要知道传入工厂类的参数,不关心创建对象的参数

简单工厂类的实现方式

直接传入判断参数key

  • Factory:
public class Factory {
	public static Product produce(String concreteProductType) {
		switch (concreteProductType) {
			case "A" :
				return new ConcreteProductA();
				break;
			case "B" :
				return new ConcreteProductB();
				break;
			default :
				throw new Exception("没有对应的产品类型");
				break;
		}
	}
}
  • 问题:
    • 如果新增产品类,需要在工厂类中新增case
    • 违背了开闭原则,这种方法是不建议使用的

利用反射

  • Factory:
public Class Factory {
	public static Product produce(String concreteProductClassPathName) throw Exception {
		try {
			Product product = (Product)Class.forName(concreteProductClassPathName).newInstance();
			return product;
		} catch (InstantiationException e) {
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			e.printStackTrace();
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		}
		throw new Exception("没有对应的产品");
	}
}
  • 问题:
    • 如果新增产品类,需要传入具体产品类的类路径名称
    • 可以通过配置文件进行优化,将具体产品类的类路径名称配置在properties文件中,通过加载配置文件将类路径名称传入工厂类的方法中
    • 这样新增产品类,只需要修改配置文件即可

反射和配置文件结合

  • product.properties:
A=com.oxford.factory.simple.ConcreteProductA
B= com.oxford.factory.simple.ConcreteProductB
  • PropertyReader: 增加一个配置文件读取类,将配置文件的信息读取到Map
public Class PropertyReader {
	public static Map<String, String> property = new HashMap<>();
	public Map<String, String> readProperty(String fileName) {
		Properties properties = new Properties();
		InputStream input = getClass.getResourceAsStream(fileName); 
		try {
			pro.load(input);
			Iterator<String> iterator = pro.StringPropertyNames().iterator();
			while (iterator.hasNext()) {
				String key = iterator.next();
				String value = properties.getProperty(key);
				map.put(key, value);
			}
			input.close();
		} catch (IOException e) {
			e.printStacTrace();
		}
		return map;
	}
}
  • Factory:
public Class Factory {
	public static Product produce(String concreteProductType) throws Exception {
		PropertyReader reader = new PropertyReder();
		Map<String, String> property = reader.readProperty("property.properties");
		try {
			Product product = (Product)Class.forName(property.get(concreteProductType)).newInstance();
			return product; 
		} catch (InstantiationException e) {
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			e.printStackTrace();
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		}
		throw new Exception("没有对应的产品");
	}
}
  • 问题:
    • 每次调用的方法时,都要解析配置文件,增加系统开销
    • 可以在文件读取类在程序启动时就加载,就可以不用在每次调用时解析配置文件了

简单工厂模式总结

  • 工厂类是整个简单工厂模式的关键:
    • 工厂类中包含必要的判断逻辑,根据给定的参数来决定创建哪一个具体的产品类
    • 通过使用工厂类,客户端只需要消费具体产品即可,而不用关注具体产品对象的创建过程
    • 通过使用简单工厂模式明确了各个类的职责,有利于整个软件体系结构的优化
  • 工厂类中集中了所有具体对象的创建逻辑,违背了高内聚的责任分配原则.这样工厂类中创建的类只能是事先考虑到的,如果需要添加新的类,则需要修改工厂类的逻辑,这违背了开闭原则
  • 当系统中的具体产品类不断增多时,就会出现要求工厂类根据不同的条件创建不同的实例的需求.这种对条件的判断和对具体产品类型的判断交错在一起,不利于对系统的扩展和维护.这样的问题可以通过使用工厂方法模式进行优化

工厂方法模式

  • 工厂方法模式Factory Method Pattern:
    • 定义一个创建对象的接口,通过实现这个接口的类来决定实例化具体的类
    • 工厂方法模式让具体的类的实例化延迟到子类中进行
  • 工厂方法模式的角色:
    • 工厂类Factory:
      • 工厂方法接口,通常返回一个抽象产品类型Product的实例对象
      • 这个类是工厂方法模式的核心,与客户端程序无关. 任何在模式中创建的具体产品都需要实现这个接口
    • 工厂实现类ConcreteFactory:
      • 工厂类接口实现,覆写工厂类Factory定义的工厂方法,返回具体产品类ConcreteProduct的抽象产品类型Product类型的实例
      • 工厂实现类ConcreteFactory中包含与客户端密切相关的逻辑,并且被客户端调用来创建具体的产品实例
    • 抽象产品类Product:
      • 工厂方法模式创建的具体产品类的父类,定义类具体产品中共有的方法
    • 具体产品类ConcreteProduct:
      • 具体产品实现类,实现了抽象产品类Product中的方法
      • 工厂模式创建的每一个对象都是具体产品类ConcreteProduct的一个实例
        Java架构师Day01-源码分析之常用设计模式_Java开发_03
  • 工厂方法模式代码实现
  • 工厂方法模式优点:
    • 在工厂方法模式中,通过工厂方法来创建客户端需要的产品ConcreteProduct, 用户只需要关心需要具体产品ConcreteProduct对应的工厂实现ConcreteFactory. 不需要关心具体产品ConcreteProduct的创建细节和具体产品类ConcreteProduct的名称
    • 基于工厂类Factory和抽象产品类Product的多态性设计是工厂方法模式的关键. 这样工厂类Factory可以自主确定需要创建何种产品ConcreteProduct的对象,并且创建具体产品ConcreteProduct对象的具体实现封装在具体工厂ConcreteFactory的内部. 具体工厂类ConcreteFactory都具有同一父类接口Factory, 因此工厂方法模式又称为多态工厂模式
    • 工厂方法模式完全符合开闭原则,有利于系统的扩展和维护. 工厂方法模式在系统中添加新产品时,只需要添加一个具体工厂类ConcreteFactory和具体产品类ConcreteProduct即可
  • 工厂方法模式缺点:
    • 工厂模式在系统中添加新产品时,需要添加具体产品类ConcreteProduct和具体工厂类ConcreteFactory, 系统中类的个数成对增加,一定程度上增加了系统复杂度以及系统编译运行的开销
  • 工厂方法模式的使用场景:
    • 一个类不需要知道所需要的对象的类: 工厂方法模式中,客户端不知道具体产品类的类名,只知道具体的产品对象由哪一个具体工厂实现来创建. 这时,客户端需要知道创建具体产品的具体工厂实现类
    • 一个类通过子类来指定创建哪一个对象: 工厂方法模式中,工厂类中只需要一个创建产品的的接口,由子类来确定具体要创建的对象,通过利用多态和里氏代换原则,可以在程序运行时,通过子类对象覆盖父类对象,从而使得系统得以扩展
    • 通过将创建具体产品的任务交由工厂类的具体工厂实现来完成,客户端不需要关心具体产品类的创建, 需要的时候动态指定产品的具体工厂实现即可. 可以将具体工厂类的类名存储在配置文件或者数据库中
    • 工厂方法模式的使用场景示例:
      • 日志记录器: 日志可以记录到本地磁盘,系统事件,远程服务器等等,用户可以选择日志记录的位置
      • 数据库访问: 当用户不知道最后系统采用哪一类数据库时,以及数据库可能会发生变化时
      • 服务器框架设计: 设计一个连接服务器的框架时,可能会用到三个协议POP3, IMAP, HTTP时,可以将三个协议看作是具体产品类,使用工厂方法模式实现

工厂方法模式总结

  • 工厂方法模式是简单工厂模式的抽象和拓展,通过多态,工厂方法模式保持了简单工厂模式的优点,改善了简单工厂模式的缺点
  • 工厂方法模式中,核心的工厂类仅仅给出具体工厂实现必须实现的接口,不再负责具体产品的创建,具体产品的创建交由具体的工厂实现完成.这样使得系统可以在不修改核心的工厂类时进行具体产品实现的扩展
  • 优点:
    • 客户端想要创建对象,只需要知道具体工厂实现即可
    • 系统的扩展性高,如果新增产品,只需要一个具体工厂实现类和具体产品类即可,符合开闭原则
    • 对客户端隐藏了具体实现,客户端只需要关心具体的工厂实现即可
  • 缺点:
    • 每次增加一个产品,都需要增加一个具体工厂实现类和具体产品类,这样使得系统中类的个数成倍增加,在一定程度上增加了系统的复杂度,也增加了系统具体类的依赖,同时类的增加也增加了编译和运行时的系统开销

抽象工厂模式

  • 抽象工厂模式Abstract Factory Pattern:
    • 提供接口或者抽象类用于创建一组相关或者相互依赖的具体产品对象,不需要指定具体的类
  • 抽象工厂模式的基本思想:
    • 工厂方法模式通过引入工厂等级结构,解决了简单工厂模式中工厂类的职责过大的问题.但是由于工厂方法模式中每个工厂只生产一类产品,这样可能会导致存在大量的工厂类的问题,这样会增加系统的开销
    • 可以将一些相关的产品组成一个产品族,由同一个工厂来统一生产
      • 产品族: 位于不同产品等级结构中功能相关联的产品组成的家族
    • 抽象工厂模式与工厂方法模式区别:
      • 抽象工厂模式:
        • 抽象工厂模式是针对多个产品的等级结构
        • 抽象工厂模式的具体产品实现或者继承于不同的接口或者抽象类
      • 工厂方法模式:
        • 工厂方法模式是针对一个产品的等级结构
        • 工厂方法模式的具体产品实现或者继承于同一个接口或者抽象类
  • 抽象工厂模式的角色:
    • 抽象工厂类AbstractFactory: 抽象工厂模式的核心,与应用的业务逻辑无关. 通常使用接口或者抽象类实现,所有具体工厂类ConcreteFactory必须实现接口或者抽象类
    • 具体工厂类ConcreteFactory: 实现工厂定义的方法,包含创建具体产品实例的业务逻辑
    • 抽象产品AbstractProduct: 定义一类产品对象的接口或者抽象类,这个类是工厂方法模式创建的对象的父类
    • 具体产品ConcreteProduct: 实现业务逻辑的具体的产品,抽象工厂中创建的每一个产品对象都是一个具体产品的实例
      Java架构师Day01-源码分析之常用设计模式_Java开发_04
  • 抽象工厂模式代码实现
  • 抽象工厂模式优点:
    • 抽象工厂模式分隔了具体类的生成,客户端不需要知道具体创建的类
    • 当一个产品族中的对象设计成一起工作时,能够保证客户端只使用同一个产品族的对象
  • 抽象工厂模式缺点:
    • 如果添加新的产品对象时,难以对产品等级结构进行扩展
  • 抽象工厂模式的使用场景:
    • 一个系统中不依赖于产品类的具体实例的创建,组合以及表达的细节
    • 系统中有多个产品族,并且每次只使用其中一种产品族
    • 同一个产品族的产品会在一起使用
    • 系统中提供一个产品类的库,客户端不依赖具体产品的实现,所有产品以同样的接口出现
    • 系统结构稳定,不会频繁增加产品族
  • 抽象工厂模式问题: 开闭原则的倾斜性
    • 抽象工厂模式中开闭原则的倾斜性是指在抽象工厂模式中,增加新的产品方便,但是增加新的产品族很麻烦
    • 开闭原则要求系统对修改关闭,对扩展开放.对于多个产品族和多个产品等级结构的系统的功能扩展增强包括:
      • 增加产品: 对于增加新的产品,只需要增加一个对应的具体工厂即可,不需要修改已有的代码
      • 增加产品族: 对于增加新的产品族的产品等级结构,需要修改所有的工厂角色,包括抽象工厂类,在所有的工厂类中都需要增加生产新产品族产品的方法,这是违背了开闭原则的
    • 因为抽象工厂模式存在开闭原则的倾斜性,因此要求在系统设计之初就要考虑整个系统的所有产品族,不会在设计完成之后再增加新的产品族,也不会删除已有的产品族.否则会导致系统有大量的修改,难以维护

抽象工厂模式总结

  • 抽象工厂模式是工厂方法模式的进一步拓展,提供了更为强大的工厂类用于系统的扩展
  • 抽象工厂模式分隔了具体类的生成,客户端无需了解产品的创建过程,这样使得很容易切换具体工厂.因为所有的具体工厂都实现了抽象工厂中定义的公共接口,因此只需要改变具体工厂的实例,就可以改变整个系统的行为
  • 当一个产品族中的多个产品对象一起工作时,可以保证客户端始终只使用同一个产品族中的对象
  • 增加新的产品很方便,无需修改已有的系统,符合开闭原则
  • 但是增加系统新的产品族的产品等级结构很麻烦,需要对原有的系统大量修改,甚至需要修改抽象层代码,违背了开闭原则
Singleton-单例模式

单例模式

  • 单例模式是一种创建模式,单例类负责自己创建自己的对象并且一个类只有一个实例对象,并且向整个系统提供这个实例.系统可以直接访问这个实例而不需要实例化
  • 单例模式的特点:
    • 单例类只有一个实例
    • 单例类必须自己创建自身的唯一实例
    • 单例类必须给其余系统对象提供创建的唯一实例

单例模式的实现方式

  • 单例模式要保证一个类只有一个实例,并且提供给全局访问,主要用于解决一个全局使用的类频繁创建和销毁的问题,通过判断系统是否存在这个单例来解决这样的问题,如果有这个单例则返回这个单例,否则就创建这个单例,只要保证构造函数是私有的即可
    • 保证一个类只有一个实例: 将该类的构造方法定义为私有方法即可
    • 提供全局一个该实例的访问点: 单例类自己创建实例,提供一个静态方法作为实例的访问点即可
  • 饿汉和懒汉比较:
    • 懒汉: 单例类对象实例懒加载,不会提前创建对象实例,只有在使用对象实例的时候才会创建对象实例
    • 饿汉: 在单例对象实例进行声明引用时就进行实例化创建对象实例
  • 单例模式除去线程不安全的懒汉,通常有五种实现方式:
    • 懒汉
    • 双检锁
    • 饿汉
    • 静态内部类
    • 枚举
  • 一般情况下,直接使用饿汉实现单例模式
  • 如果明确要求懒加载通常使用静态内部类实现单例模式
  • 如果有关于反序列化创建对象会考虑使用枚举实现单例模式
  • 静态类Static :
    • 静态类在第一次运行时直接初始化,也不需要在延迟加载中使用
    • 在不需要维持任何状态,仅仅用于全局访问时,使用静态类的方式更加方便
    • 如果需要被继承或者需要维持一些特定状态下的情况,就适合使用单例模式

线程不安全懒汉

线程安全懒汉

  • 单例模式线程安全懒汉Singleton示例
  • 解决了多线程环境下创建多个实例的问题
  • 存在每次获取实例都需要申请锁的问题,方法效率低下,因为在任何时候只能有一个线程可以调用getInstance() 方法

双检锁

  • 双重检查锁模式: doule checked locking pattern
    • 使用同步块加锁的方法
    • 会有两次检查instance == null
      • 一次在同步块外
      • 一次在同步块内
        • 因为会有多个线程一起进入同步块外的if
        • 如果不在同步块内不进行二次检验就会导致生成多个实例
  • 单例模式双检锁Singleton示例
  • volatile:
    • 对于计算机中的指令而言 ,CPU和编译器为了提升程序的执行效率,通常会按照一定的规则对指令进行优化
    • 如果两条指令互不依赖,那么指令执行的顺序可能不是源码的编写顺序
    • 形如instance = new Instance() 方法创建实例执行分为三步:
      • 分配对象内存空间: 给新创建的Instance对象分配内存
      • 初始化对象: 调用单例类的构造函数来初始化成员变量
      • 设置instance指向新创建的对象分配的内存地址,此时instance != null
        • 因为上面的初始化对象和设置instance指向新创建的对象分配的内存地址不存在数据上的依赖关系,无论哪一步先执行都不会影响最终结果,所以程序在编译时,顺序就会发生改变:
          • 分配对象内存空间
          • 设置instance指向新创建对象分配的内存地址
          • 初始化对象
        • CPU和编译器在指令重排时,不会关心指令重排执行是否影响多线程的执行结果. 如果不加volatile关键字,如果有多个线程访问getInstance() 方法时,如果刚好发生了指令重排,可能会出现以下情况:
          • 当第一个线程获取锁并且进入到第二个if方法后,先分配内存空间,然后instance指向刚刚分配的内存地址,此时instance不等于null. 但是此时instance还没有初始化完成
          • 如果此时有另一个线程调用getInstance() 方法,在第一个if的判断时结果就为false, 就会直接返回没有初始化完成的instance, 这样可能会导致程序NPE异常
    • 使用volatile的原因是禁止指令重新排序:
      • volatile变量进行赋值操作后会有一个内存隔离
      • 读操作不会重排序到内存隔离之中
      • 比如在上面操作中,读操作必须在执行完1,2,3或者1,3,2步骤之后才会执行读取到结果,否则不会读取到相关结果

饿汉

  • 单例模式饿汉Singleton示例
  • 优点:
    • 在单例类中,装载类的时候就创建对象实例.因为单例类的实例声明为staticfinal变量,在第一次加在类到内存中时就会初始化,所以创建实例本身时线程安全的
  • 缺点:
    • 饿汉模式不是一种懒加载模式,即便客户端没有调用getInstance() 方法,单例类也会在类第一次加载时初始化
    • 使用饿汉模式创建单例类实例在某些场景中无法使用:
      • 比如因为饿汉创建的实例声明为final变量
      • 如果单例类Singleton的实例的创建依赖参数或者配置文件
      • 需要在getInstance() 方法之前调用方法为单例类的实例设置参数,此时这种饿汉模式就无法使用

静态内部类

  • 单例模式静态内部类Singleton示例
  • 使用静态内部类模式创建单例类实例是使用JVM机制保证线程安全:
    • 静态单例对象没有作为单例类的成员变量直接实例化,所以当类加载时不会实例化单例类
    • 第一次调用getInstance() 方法时将加载静态内部类Nest. 在静态内部类中定义了一个static类型的变量instance, 这时会首先初始化这个变量
    • 通过JVM来保证线程安全,确保该成员变量只初始化一次
    • 由于getInstance() 方法并没有加线程锁,所以对性能没有什么影响
  • 静态内部类的优点:
    • 静态内部类Nest是私有的,只能通过getInstance() 方法进行访问,所以这是懒加载的
    • 读取实例时不会进行同步锁的获取,性能较好
    • 静态内部类不依赖JDK版本

枚举

  • 单例模式枚举Singleton示例
  • 使用枚举方式实现单例的最大特点是非常简单
  • 可以通过Enum.INSTANCE来访问实例,和getInstance() 方法比较更加简单
  • 枚举的创建默认就是线程安全的方法,而且能防止反射以及反序列化导致重新创建新的对象
    • Enum类内部使用Enum类型判定防止通过反射创建新的对象
    • Enum类通过对象的类型和枚举名称将对象进行序列化,然后通过valueOf() 方法匹配枚举名称找到内存中的唯一对象实例,这样可以防止反序列化时创建新的对象
  • 懒汉式和饿汉式实现的单例模式破坏 : 无论是通过懒汉式还是饿汉式实现的单例模式,都可能通过反射和反序列化破坏掉单例的特性,可以创建多个对象
  • 反射破坏单例模式: 利用反射,可以强制访问单例类的私有构造器,创建新的对象
public static void main(String[] args) {
	// 利用反射获取单例类的构造器
	Constructor<Singleton> constructor = Singleton.class.getDeclaredConstructor();
	// 设置访问私有构造器
 	constructor.setAccessiable(true);
 	// 利用反射创建新的对象
 	Singleton newInstance = constructor.newInstance();
 	// 通过单例模式创建单例对象
 	Singleton singletonInstance = Singleton.getInstance();
 	// 此时这两个对象是两个不同的对象,返回false
 	System.out.println(singletonInstance  == newInstance);
}
  • 反序列化破坏单例模式: 通过readObject() 方法读取对象时会返回一个新的对象实例
public static void main(String[] args) {
	// 创建一个输出流对象
	ObjectOutputStream os = new ObjectOutputStream(new FileOutputStream("Singleton.file"));
	// 将单例类对象写入到文件中
	Singleton singletonInstance = Singleton.getInstance();
	os.writeObject(singleton);
	// 从文件中读取单例对象
	File file = new File("Singleton.file");
	ObjectInputStream is = new ObjectInputStream(new FileInputStream(file));
	Singleton newInstance = (Singleton)is.readObject();
	// 此时这两个对象是两个不同的对象,返回false
	System.out.println(singletonInstance == newInstance);
}
Delegate-委托模式

委托模式

  • 委托模式: 软件设计模式中的一项基本技巧,不属于23种常用的设计模式
    • 在委托模式中,有两个对象参与处理同一个请求
    • 接收请求的对象将请求委托给另一个对象来处理
  • 委托模式使得项目可以使用聚合来代替继承
  • 委托模式的工作方式:
    • 委托模式的委托类中包含被委托类的实例的引用,实现被委托类的接口
    • 委托类将被委托类方法的调用转发给委托类的内部的实例引用
  • 委托模式的角色:
    • 委托者Delegator :
      • 保存被委托者Delegate的实例引用
      • 实现被委托者Delegate的接口
      • 将对被委托者Delegate接口方法的调用转发给委托者Delegator
    • 被委托者Delegate :
      • 接收委托者Delegator的调用
      • 实现委托者Delegator的接口

委托模式示例

简单委托模式

  • 被委托者Delegate :
public class Delegate {
	/**
	 * 被委托者内部方法
	 */
	public void doDelegate() {
		System.out.println("被委托者执行自己的方法...");
	}
}
  • 委托者Delegator :
public class Delegator {
	// 保存创建一个被委托者
	Delegate delegate = new Delegate();
	
	/**
	 * 委托者相关方法
	 */
	 public void doDelegator() {
	 	// 内部请求被委托者的方法
		delegate.doDelegate();
	 }
}
  • 客户端DelegateTest :
public class DelegateTest {
	public static void main(String[] args) {
		Delegator delegator = new Delegator();
		// 执行委托者方法,实现的是被委托者的处理逻辑
		delegator.doDelegator();
	}
}

复杂委托模式

  • 复杂委托模式的一个示例是业务委托模式Business Delegate Pattern用于对表示层和业务层解耦,基本上用来减少通信或者对表示层代码中的业务层代码的远程查询功能,包含以下四个角色:

    • 业务服务Business Service: 业务服务接口. 实现该业务服务的实体类,提供了实际的业务实现逻辑
    • 查询服务Lookup Service: 查找服务对象负责获取相关的业务实现,并提供业务对象对业务委托对象的访问
    • 业务委托类Business Delegate: 为客户端实体提供的入口类,提供了对业务服务员方法的访问
    • 客户端Client: 表示层. 可以是JSP,servlet或者Java UI等代码
  • 委托者Delegator可以选择委托被委托者DelegateA或者被委托者DelegateB. 委托者Delegator中有选择被委托者Delegate的方法.因为被委托者Delegate都需要实现接口Interface规定的方法,所以这里的委托模式是线程安全的

  • 接口Interface :

public interface Interface {
	void doDelegate();
}
  • 被委托者DelegateA :
public class DelegateA implements Interface {
	@Override
	public void doDelegate() {
		System.out.println("被委托者A实现委托方法...");
	}
} 
  • 被委托者DelegateB :
public class DelegateB implements Interface {
	@Override
	public void doDelegate() {
		System.out.println("被委托者B实现委托方法...");
	}
}
  • 委托者Delegator :
public class Delegator implements Interface {
	// 默认委托被委托者DelegateA
	Interface delegate = new DelegateA();
	
	@Override
	public void doDelegate() {
		delegate.doDelegate();
	}
	
	/**
	 * 委托A
	 */
	public void delegateA() {
		delegate = new DelegateA();
	}

	public void delegateB() {
		delegate = new DelegateB();
	}
}
  • 客户端DelegateTest :
public class DelegateTest {
	public static void main(String[] args) {
		Delegator delegator = new Delegator();
		delegator.doDelegate();

		delegator.delegateB();
		delegator.doDelegate();
	}
} 

委托模式和代理模式比较

委托模式

  • 委托模式是将某个对象的请求委托给另一个对象,这种委托是将整个操作全权委托
  • 委托模式设计的出发点是使用组合代理继承.在面向对象的软件开发中,要多用组合少用继承

代理模式

  • 代理模式的目的在于提供一种代理控制对对象的访问
Strategy-策略模式

策略模式

  • 策略模式: 定义一组算法,将每一个算法封装到具有共同接口的独立类中,从而使得各个算法之间可以相互替换
    • 策略模式属于行为型模式,可以使得算法在不影响客户端的情况下自由切换
    • 策略模式是对算法的包装,将使用算法和算法本身分离开来,委托给不同的对象进行管理
    • 策略模式通常将一个系列的算法包装到一系列的策略类中,作为一个抽象策略类的子类
  • 策略模式应用场景:
    • 当实现某一个功能存在多个算法或者策略时,可以根据环境或者条件的不同选择不同的算法或者策略来实现某个功能
      • 一个系统需要动态地在几种算法中选择一种时,可以将每个算法封装到策略类中
      • 一个类定义了多种行为,并且这些行为在这个类的操作中以多个条件语句的形式出现,可以将每个条件分支移入各自策略类中来代替这些条件语句
      • 系统中各个算法策略彼此独立,并且要求对客户端隐藏具体算法的实现细节
      • 系统要求使用算法的客户不应知道操作的数据,可以使用策略模式隐藏与算法有关的数据结构
      • 多个类的区别只是表现行为的不同,可以使用策略模式在系统运行时选择需要具体执行的行为
    • 比如数据排序策略的冒泡排序,选择排序,插入排序和二叉树排序等
      Java架构师Day01-源码分析之常用设计模式_Java教程_05
  • 策略模式包含三个角色:
    • 环境Context: 持有一个抽象策略Strategy的引用
    • 抽象策略Strategy: 抽象角色.给出所有具体策略类所需的接口,通常是由一个接口或者抽象类实现
    • 具体策略ConcreteStrategy: 具体策略类. 封装了具体的算法
  • 策略模式Strategy示例

策略模式总结

  • 策略模式的重点:
    • 不是如何实现算法
    • 而是在于如何组织,调用一系列的算法,从而让程序的结构更加灵活,具有良好的可维护性以及扩展性
  • 算法的平等性:
    • 策略模式的一个特点在于各个算法策略的平等性
    • 对于一系列算法之间是平等的,只有这样的算法之间才可以相互替换
    • 策略模式中具体策略算法在实现上是相互独立的,相互之间没有依赖
  • 运行策略的唯一性:
    • 在程序运行期间,策略在某一个时间点只能使用一个具体的策略实现对象
    • 策略模式可以动态地在不同的具体算法策略之间进行切换,但在同一个时间点只能使用一个具体算法策略
  • 策略模式中公有的行为:
    • 通常情况下,策略模式的各个具体策略之间会有一些公有的行为
    • 此时,应该将这些各个具体策略之间公有的行为放到共同的抽象策略Strategy类中
    • 此时,抽象策略Strategy类可以使用抽象类Abstract实现,而非必须使用接口

策略模式优点

  • 策略模式提供了管理一系列算法的方法. 策略类Strategy的等级结构定义了一个算法或者是行为族,可以通过继承将公有的代码放到父类Strategy中,从而避免代码重复
  • 策略模式避免了使用多重条件if-else语句,提高了项目的可维护性
  • 策略模式提供相同行为的不同实现,客户端可以根据不同的时间和空间要求选择不同的实现策略
  • 策略模式完美支持开闭原则,可以在不修改原代码的情况下,灵活增加新的算法策略
  • 策略模式将算法的使用放到环境类中,算法的实现放到具体策略类中,实现了算法的使用和算法的实现的二者分离

策略模式缺点

  • 策略模式只适用于客户端知道具体算法策略或者行为的情况. 客户端必须知道所有的策略类,并自行决定使用具体的策略类,也就是说客户端必须理解这些算法的区别来决定选择使用什么样的算法
  • 策略模式会将每一个具体算法策略单独封装成一个类,如果存在大量具体算法策略时,会造成系统中存在大量的对象

策略工厂模式

  • 在使用策略模式的系统中,当一个系统的存在大量的策略时,客户端管理的策略算法将很复杂,可以通过在环境类中使用策略工厂模式来管理这些策略类
    Java架构师Day01-源码分析之常用设计模式_Java开发_06
Prototype-原型模式
  • 原型模式: 通过原型的实例来指定创建对象的类型,然后通过拷贝来创建更多同类型的对象
  • 原型模式的特点:
    • 原型模式属于创建型模式
    • 原型模式要求原型类实现拷贝自身的接口,这样就可以通过拷贝原型的实例来创建新的对象
    • 原型模式拷贝原型实例创建的新对象不需要关心对象本身的类型,只需要实现原型实例拷贝自身的方法,就可以通过自定义的拷贝方法来创建新的对象,不需要通过new创建新的对象

原型模式的实现方式

  • 原型模式主要用于对象的复制,核心就是抽象原型Prototype
  • 抽象原型Prototype需要具备两个条件:
    • 实现Clonable接口:
      • Clonable接口的作用是在运行时通知JVM可以安全地在实现Clonable接口的类上使用clone方法
      • JVM中,只有实现了Clonable接口的类才可以拷贝,否则会抛出CloneNotSupportedException异常
    • 重写Object类中的clone方法:
      • 所有类的父类都是Object,Object类中有clone() 方法用于返回一个拷贝的对象,但是方法的作用域是protected, 一般类无法进行调用
      • 抽象原型Prototypeclone() 方法的作用域修改为public类型
  • 原型模式的适用场景:
    • 在需要重复创建相似的对象时可以使用原型模式,即只是个别的几个属性不同时
    • 直接创建对象的成本损耗大.比如初始化时间长,CPU占用大,网络资源占用过多,需要优化资源等
    • 直接创建一个对象后还需要准备大量的数据以及访问权限时,需要提高性能或者安全性
    • 系统中大量使用该类对象,且各个调用者都需要给对象的属性重新赋值
  • 原型模式有两种实现方式:
    • 简单实现方式
    • 登记实现方式

简单实现方式

Java架构师Day01-源码分析之常用设计模式_Java开发_07

  • 原型模式的简单实现方式包含三个角色:
    • 抽象原型Prototype: 抽象原型类.给出所有的具体原型所需要实现的拷贝方法的接口或者抽象类
    • 具体原型ConcretePrototype: 用于拷贝的原型实例.实现了抽象原型Prototype所要求的接口
    • 客户Client: 客户端Client用来请求创建新的对象

原型管理器实现方式

Java架构师Day01-源码分析之常用设计模式_Java开发_08

  • 原型模式可以扩展成为带有原型管理器的原型模式:
    • 在原型模式的基础上增加了新的原型管理器PrototypeManager
    • 原型管理器PrototypeManager类中使用HashMap保存多个复制的原型对象
    • 客户端Client类可以通过原型管理器中的get(String id) 方法获取复制的原型对象

原型模式注意点

  • 原型模式不会调用类的构造方法:
    • 原型模式通过Object类的clone() 方法拷贝对象,是直接在内存中拷贝原型实例的数据,所以不会调用类的构造方法
    • 类的构造方法不会执行并且访问权限对原型模式无效.因为单例模式中的将类的构造方法设置为private类型,但是clone() 方法是无视构造方法的权限的.因此原型模式是与单例模式相互冲突的,所以要特别注意原型模式不可以和单例模式一起使用
  • 深拷贝与浅拷贝问题:
    • 原型模式通过Object类的clone() 方法拷贝对象时只会拷贝对象的基本数据类型以及对应的封装类型和值类型的String类型,不会拷贝数组,容器对象,引用对象等.这就是浅拷贝
    • 如果要实现深拷贝,需要将原型模式中的数组,容器对象和引用对象另行拷贝

原型模式总结

  • 原型模式本省比较简单,易于实现.主要是重写Object中的clone() 方法,根据要求实现浅拷贝或者是深拷贝
  • 原型模式的重点在浅拷贝和深拷贝的理解和运用

原型模式优点

  • 使用原型模式创建对象比直接使用new创建对象的性能要好: 因为Object中的clone() 方法是一个本地native方法,直接操作内存中的二进制流.尤其在拷贝大对象时,性能提升明显
  • 使用原型模式简化了对象的创建: 通过使用原型模式可以让对象的创建像复制粘贴一样简单
  • 可以使用深拷贝的方式保存对象的状态: 通过使用原型模式拷贝对象,并将状态保存起来,可以简化对象的创建过程,以便在需要时使用,可以辅助实现撤销的操作

原型模式缺点

  • 原型模式需要为每一个类配置一个clone() 方法
  • clone() 方法在每个类的内部,当对已有类进行重构时,需要修改代码,违背了开闭原则
  • 实现深拷贝较为复杂,而且当对象之间存在多重嵌套时,为了实现深拷贝,需要每一层对象对应的类都必须支持深拷贝,实现比较复杂
Template-模板模式

模板方法模式

  • 模板方法模式: 使用一个抽象类定义算法模板结构,将某些具体算法步骤的实现延迟到子类中进行,使得子类在不改变算法结构的情况下重新定义某些算法的具体实现.算法的调用在抽象类中进行
  • 模板方法模式是行为型模式, 基于继承的代码复用
  • 模板方法的关键: 子类可以置换父类中的可变部分,但是不可以改变模板方法中实现的顶级逻辑
    • 在模板方法模式中定义新的子类时,不是按照控制流程,而是按照责任的思想
    • 要考虑子类中对父类中的哪些方法是必须要置换的,哪些是可以置换的,哪些是不能置换的. 通过模板方法可以使得这些责任清晰明朗
  • 模板方法模式的应用场景:
    • 重要的,复杂的方法可以作为模板方法
    • 使用一个类来实现一个算法的不变的逻辑,并将可变的逻辑延迟到子类中进行实现
    • 各个子类中的公共行为提取集中到一个父类中,用来避免代码重复
    • 控制子类的扩展

模板方法模式示例

Java架构师Day01-源码分析之常用设计模式_Java开发_09

  • 模板方法模式包含两个角色:
    • 抽象模板AbstractTemplate:
      • 定义抽象操作的算法模板结构,用于在子类中实现
        • 抽象操作也就是基本操作,是一个顶级逻辑的组成步骤
      • 定义并实现一个模板方法
        • 模板方法是一个具体方法
        • 给出了一个顶级逻辑的结构,而逻辑结构的组成步骤在相应的抽象操作中,延迟到子类中实现
        • 顶级逻辑结构中也可能会调用一些具体方法
    • 具体模板ConcreteTemplate:
      • 实现抽象模板AbstractTemplate中定义的抽象方法.是一个顶级逻辑的组成步骤
      • 每一个抽象模板AbstractTemplate都可以有任意多个具体模板ConcreteTemplate相对应. 每一个具体模板ConcreteTemplate都可以对这些抽象模板AbstractTemplate中顶级逻辑的组成步骤进行不同的实现,从而使得顶级逻辑的实现各有不同
  • 模板方法模式中的方法: 模板方法和基本方法
    • 模板方法:
      • 模板方法定义在抽象类中,将多个基本方法操作组合成一个总的算法或者行为
      • 一个抽象类可以包含任意多个模板方法, 每一个模板方法可以调用任意多个具体方法
    • 基本方法:
      • 抽象方法AbstractMethod : 抽象方法由抽象类声明,由具体子类实现
      • 具体方法ConcreteMethod : 具体方法由抽象类声明并实现,子类不进行实现或者替换
      • 钩子方法HookMethod : 钩子方法由抽象类声明并实现,子类可以进行扩展.
        • 一般情况下,抽象类会给出钩子方法的一个空实现作为默认实现
        • 钩子方法使得子类不需要给出父类的所有方法实现,因为一个具体的子类通常不需要实现父类中的所有的方法
  • 模板方式模式实现:
    • 分析目标算法,确定能否分解为多个步骤.从各个子类的角度,考虑哪些步骤能够通用,哪些步骤有不同
    • 创建一个抽象类并声明一个模板方法和代表算法步骤的一系列的抽象方法. 在模板方法中根据算法的结构依次调用算法步骤,使用final修饰符修饰模板方法以防止子类对模板方法进行重写
    • 将子类的共有的方法放到父类中,子类中只需要重写可变的算法逻辑
    • 可以考虑在父类中添加钩子方法,默认为空实现,用于子类对算法逻辑进行扩展
    • 每一个算法子类都需要新建一个具体的子类,必须实现所有的抽象步骤,可以重写部分可选的扩展步骤
  • 模板方法模式Template Method示例

模板方法模式总结

模板方式模式注意点

  • 通常情况下,模板方法都要加上final关键字

模板方法模式优点

  • 实现反向控制: 父类调用子类的操作,通过子类的具体实现扩展成不同的行为,实现了反向控制,符合开闭原则
  • 提高代码的复用性: 将共有的代码放到抽象的父类中,不同的代码实现放在子类中
  • 提高了系统的扩展性: 将不同的代码放到不同的子类中,通过对子类的扩展增加新的行为
  • 行为由父类控制,子类实现

模板方法模式缺点

  • 增加了系统的复杂性: 引入了抽象类,每一个不同的实现都需要一个子类来实现,导致系统中类的个数增加
  • 通过子类修改父类的默认步骤实现,可能违反了里氏代换的原则

模板方法模式和工厂模式,的区别

模板方法模式

  • 模板方法模式基于继承机制,允许在子类中扩展和修改算法相关步骤实现
  • 模板方法模式是静态的,是在类层次上进行的

工厂方法模式

  • 工厂方法模式是一种特殊的模板方法模式
  • 工厂方法可以作为一个大型的模板方法模式中的一个步骤

策略模式

  • 策略模式基于组合机制,可以通过对相应的行为提供不同的策略来改变对象的部分行为
  • 策略模式是动态的,可以在运行时切换行为.策略模式是在对象层次上进行的