设计模式
一、单例模式
饿汉式
1 | public class Singleton { |
实现方式:在类加载的时候就对实例进行创建,实例在整个程序周期都存在。
优点:避免了多线程同步问题
缺点:不需要的时候就加载,会造成资源浪费
懒汉式
1 | public class Singleton{ |
实现方式:在需要的时候才会创建对象。如果对象已经被创建则不会再创建对象
优点:资源消耗少,
缺点:线程不安全,需要加同步方法。解决方法:加锁解决线程同步问题
1. 加synchronized关键字
1 | public class Singleton { |
实现方式:在调用方法上面加锁机制,实现同步操作
优点:解决线程安全问题
缺点:性能消耗损失较大
2. 用“双重检查加锁”
1 | public class Singleton { |
实现方式:不在每次调用的时候都进行同步加锁操作,只需要在第一次调用的时候进行加锁操作。并且通过volatile来保证数据的可见性
优点:保证了线程安全,并且降低了消耗
缺点:对java版本有要求。1.5以下版本用volatile可能会出现问题
volatule关键字详解:https://zhuanlan.zhihu.com/p/138819184
3. 用“静态内部类”
1 | public class Singleton { |
实现方式:在调用方法里面添加静态内部类
优点:外部类在加载的时候并不会立即加载内部类。性能好,无需加锁
缺点:由JVM类加载保证
二、工厂模式
简单工厂模式
1 | public interface Moveable{ |
抽象工厂模式
1 | /* test*/ |
模拟Spring工厂实现
1 | public static void main(String[] args) throws Exception{ |
模拟 IOC:
- 都说
Spring
是个bean
容器, 以下的代码将展示它是如何生成bean
, 并把bean
放入容器中供用户获取的. - 思路比较简单:
- 创建
BeanFactory
工厂接口, 添加方法getBean()
. - 创建
BeanFactory
的实现类ClassPathXmlApplicationContext
. 将在该实现类中展示IOC
的具体实现. ClassPathXmlApplicationContext
需要一个container
容器存放创建的bean
对象, 这里使用HashMap
实现.- 在
ClassPathXmlApplicationContext
的构造方法中读取spring
的配置文件, 这里使用到了dom4j
. 读取配置文件后根据bean
的class
属性使用反射创建出bean
对象. 然后把id
和bean
对象分别作为key
和value
添加到容器中. - 当工厂被调用
getBean()
方法时, 从容器中找到对应的bean
并返回.
三、代理模式
分类:
分类:
静态代理: 在编译时就已经实现,编译完成后代理类是一个实际的class文件
动态代理: 在运行时动态生成的,即编译完成后没有实际的class文件,而是在运行时动态生成类字节码,并加载到JVM中
静态代理
1 | public interface UserDao { |
使用JDK静态代理很容易就完成了对一个类的代理操作。但是JDK
静态代理的缺点也暴露了出来:由于代理只能为一个类服务,如果需要代理的类很多,那么就需要编写大量的代理类,比较繁琐
JDK动态代理
使用JDK动态代理的五大步骤:
- 通过实现InvocationHandler接口来自定义自己的InvocationHandler;
- 通过
Proxy.getProxyClass
获得动态代理类; - 通过反射机制获得代理类的构造方法,方法签名为
getConstructor(InvocationHandler.class)
; - 通过构造函数获得代理对象并将自定义的
InvocationHandler
实例对象传为参数传入; - 通过代理对象调用目标方法;
1 | public interface IHello { |
JDK静态代理与JDK动态代理之间有些许相似,比如说都要创建代理类,以及代理类都要实现接口等。
不同之处: 在静态代理中我们需要对哪个接口和哪个被代理类创建代理类,所以我们在编译前就需要代理类实现与被代理类相同的接口,并且直接在实现的方法中调用被代理类相应的方法;但是动态代理则不同,我们不知道要针对哪个接口、哪个被代理类创建代理类,因为它是在运行时被创建的。
一句话来总结一下JDK静态代理和JDK动态代理的区别:
JDK静态代理是通过直接编码创建的,而JDK
动态代理是利用反射机制在运行时创建代理类的。
其实在动态代理中,核心是InvocationHandler
。每一个代理的实例都会有一个关联的调用处理程序(InvocationHandler)。对待代理实例进行调用时,将对方法的调用进行编码并指派到它的调用处理器(InvocationHandler)的invoke
方法
对代理对象实例方法的调用都是通过InvocationHandler中的invoke方法来完成的,而invoke方法会根据传入的代理对象、方法名称以及参数决定调用代理的哪个方法。
CGLIB动态代理
CGLIB包的底层是通过使用一个小而快的字节码处理框架ASM
,来转换字节码并生成新的类
CGLIB代理实现如下:
- 首先实现一个MethodInterceptor,方法调用会被转发到该类的intercept()方法。
- 然后在需要使用的时候,通过CGLIB动态代理获取代理对象。
1 | public class HelloService { |
JDK代理要求被代理的类必须实现接口,有很强的局限性。
而CGLIB动态代理则没有此类强制性要求。简单的说,CGLIB
会让生成的代理类继承被代理类,并在代理类中对代理方法进行强化处理(前置处理、后置处理等)。
总结一下CGLIB在进行代理的时候都进行了哪些工作
- 生成的代理类继承被代理类。在这里我们需要注意一点:如果委托类被final修饰,那么它不可被继承,即不可被代理;同样,如果委托类中存在final修饰的方法,那么该方法也不可被代理
- 代理类会为委托方法生成两个方法,一个是与委托方法签名相同的方法,它在方法中会通过
super
调用委托方法;另一个是代理类独有的方法 - 当执行代理对象的方法时,会首先判断一下是否存在实现了
MethodInterceptor
接口的CGLIB$CALLBACK_0
;,如果存在,则将调用MethodInterceptor
中的intercept
方法
在intercept
方法中,我们除了会调用委托方法,还会进行一些增强操作。在Spring AOP中,典型的应用场景就是在某些敏感方法执行前后进行操作日志记录
在CGLIB中,方法的调用并不是通过反射来完成的,而是直接对方法进行调用:通过FastClass机制对Class对象进行特别的处理,比如将会用数组保存method的引用,每次调用方法的时候都是通过一个index下标来保持对方法的引用
Fastclass机制
CGLIB采用了FastClass的机制来实现对被拦截方法的调用。
FastClass机制就是对一个类的方法建立索引,通过索引来直接调用相应的方法
1 | public class test10 { |
三种代理方式之间对比
CGlib比JDK快?
- 使用CGLiB实现动态代理,CGLib底层采用ASM字节码生成框架,使用字节码技术生成代理类, 在jdk6之前比使用Java反射效率要高。唯一需要注意的是,CGLib不能对声明为final的方法进行代理, 因为CGLib原理是动态生成被代理类的子类。
- 在jdk6、jdk7、jdk8逐步对JDK动态代理优化之后,在调用次数较少的情况下,JDK代理效率高于CGLIB代理效率。只有当进行大量调用的时候,jdk6和jdk7比CGLIB代理效率低一点,但是到jdk8的时候,jdk代理效率高于CGLIB代理,总之,每一次jdk版本升级,jdk代理效率都得到提升,而CGLIB代理消息确有点跟不上步伐。
Spring如何选择用JDK还是CGLIB?
- 当Bean实现接口时,Spring就会用JDK的动态代理。
- 当Bean没有实现接口时,Spring使用CGlib实现。
- 可以强制使用CGlib
- 本文作者: GHOSTLaycoo
- 本文链接: http://example.com/2021/12/18/设计模式/
- 版权声明: 本博客所有文章除特别声明外,均采用 MIT 许可协议。转载请注明出处!