aop是指面向切面编程,ProxyFactoryBean是spring aop的底层实现与源头,为什么这么说呢?首先我们看一段配置:

1、target是目标对象,需要对其进行切面增强

2、proxyInterfaces是指代理对象所实现的接口

3、interceptorNames:是指通知器(Advisor)列表,通知器中包含了通知advice与切点pointcut

概括一下,ProxyFactoryBean的作用是:

针对目标对象来创建代理对象,将对目标对象方法的调用转到对相应代理对象方法的调用,并且可以在代理对象方法调用前后执行与之匹配的各个通知器中定义好的方法

spring通过两种方式来创建目标代理对象:

1、JDK动态代理

2、CGLIB

在spring中这两种代理方式都实现了AopProxy接口去,如下图所示:

spring中AopProxyFactory是一个接口,接口中只有一个方法createAopProxy,这个接口是有一个实现类DefaultAopProxyFactory,用来创建AopProxy对象,如下图所示:

下面我们比较并实现以下两种代理方式:

首先明确以下我们常用的代理模式有什么问题,在我们经常使用的代理模式中,真实对象必须是事先存在的,并将其作为代理对象的内部属性,在使用时,一个真实角色必须对应一个dialing角色,如果大量使用会导致类的急剧膨胀,java的动态代理类就可以解决这个问题。

JDK动态代理

java动态代理类位于java.lang.reflect包下,一般主要涉及到两个类:

1、Interface InvocationHandler:该接口中只定义了一个方法

 public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable;

第一个参数proxy是指代理类,method是被代理的方法,args为该方法的参数数组

2、Proxy:该类为动态代理类,其主要方法如下

public static Object newProxyInstance(ClassLoader loader,
Class<?>[] interfaces,
InvocationHandler h)
throws IllegalArgumentException

该方法返回代理类的一个实例,返回后的代理类可以被当作代理类使用

JDK动态代理实现步骤:

1、创建一个实现接口InvocationHandler的类(DynamicSubject),它必须实现invoke方法

public class DynamicSubject implements InvocationHandler {

    private Object sub;

    public DynamicSubject(Object sub) {
this.sub = sub;
} public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("before calling:" + method);
method.invoke(sub, args);
System.out.println(args == null);
System.out.println("after calling:" + method);
return null;
}
}

2、创建被代理的类(RealSubject)以及接口(Subject)

public interface Subject {
public void request();
}
public class RealSubject implements Subject {

    public void request() {
System.out.println("from real subject");
}
}

3、通过Proxy的静态方法newProxyInstance创建一个代理

public class Client {

    public static void main(String[] args) {

        RealSubject realSubject = new RealSubject();
InvocationHandler handler = new DynamicSubject(realSubject);
Class<?> classType = handler.getClass();
Subject subject = (Subject) Proxy.newProxyInstance(classType.getClassLoader(), realSubject.getClass().getInterfaces(), handler);
subject.request();
System.out.println(subject.getClass());
}
}

如果目标类并未实现接口,那么Spring就会使用CGLIB库为其创建代理,如下demo运用了CGLIB实现代理

1、创建被代理类Person

public class Person {

    public void study() {
System.out.println("study");
}
}

2、创建CglibProxy实现代理

public class CglibProxy implements MethodInterceptor {

    private Enhancer enhancer = new Enhancer();

    public Object getProxy(Class<?> clazz) {
enhancer.setSuperclass(clazz);
enhancer.setCallback(this);
Object object = enhancer.create();
return object;
} public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable { System.out.println(o.getClass().getName() + "." + method.getName()); //执行父类对应方法
final Object result = methodProxy.invokeSuper(o, objects); System.out.println("执行结束"); return result;
} public static void main(String[] args) { CglibProxy proxy = new CglibProxy();
Person person = (Person) proxy.getProxy(Person.class);
System.out.println(person.getClass().getName());
person.study(); }
}

下面我们来比较一下,两种代理方法的区别

1、JDK动态代理只能对实现了接口的类生成代理,而不能针对类

2、CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法

3、在spring中如果bean实现了接口spring用JDK动态代理,如果bean没有实现接口,spring使用CGLIB实现代理

4、CGLIb不能对声明为final的方法进行dialing,因为CGLib原理是动态生成被代理类的子类

5、JDK动态代理通过反射的newInstance方法产生代理类的对象

spring源码解读-aop的更多相关文章

  1. Spring 源码学习——Aop

    Spring 源码学习--Aop 什么是 AOP 以下是百度百科的解释:AOP 为 Aspect Oriented Programming 的缩写,意为:面向切面编程通过预编译的方式和运行期动态代理实 ...

  2. Spring源码之AOP的使用

    Spring往期精彩文章 Spring源码搭建 Spring源码阅读一 前言 我们都知道Java是一门面向对象(OOP)的语言,所谓万物皆对象.但是它也存在着一些个弊端:当你需要给多个不具有继承关系的 ...

  3. Spring源码解读之BeanFactoryPostProcessor的处理

    前言 前段时间旁听了某课堂两节Spring源码解析课,刚好最近自己又在重新学习中,便在这里记录一下学习所得.我之前写过一篇博文,是介绍BeanFactoryPostProcessor跟BeanPost ...

  4. Spring源码解读--(一)源码下载

    走在Java程序员这条路上,网上Java各种工具满天飞,写个简单的CRUD,相信是个开发都能写出来,于是在思考如何可以在同行业中更有竞争力(其实就是如何赚更多钱).那么,老大给我推荐了Spring源码 ...

  5. 【Spring源码解读】bean标签中的属性

    说明 今天在阅读Spring源码的时候,发现在加载xml中的bean时,解析了很多标签,其中有常用的如:scope.autowire.lazy-init.init-method.destroy-met ...

  6. Spring源码解读:核心类DefaultListableBeanFactory的继承体系

    1 简介 我们常用的ClassPathXmlApplicationContext是AbstractRefreshableApplicationContext的子类,而DefaultListableBe ...

  7. Spring源码解读(二):Spring AOP

    一.AOP介绍 面向方面编程(AOP)通过提供另一种思考程序结构的方式来补充面向对象编程(OOP).OOP中模块化的关键单元是类,而在AOP中,模块化单元是方面.方面实现了诸如跨越多种类型和对象的事务 ...

  8. 专治不会看源码的毛病--spring源码解析AOP篇

    昨天有个大牛说我啰嗦,眼光比较细碎,看不到重点.太他爷爷的有道理了!要说看人品,还是女孩子强一些.原来记得看到一个男孩子的抱怨,说怎么两人刚刚开始在一起,女孩子在心里就已经和他过完了一辈子.哥哥们,不 ...

  9. Spring源码解读Spring IOC原理

    一.什么是Ioc/DI? IoC 容器:最主要是完成了完成对象的创建和依赖的管理注入等等. 先从我们自己设计这样一个视角来考虑: 所谓控制反转,就是把原先我们代码里面需要实现的对象创建.依赖的代码,反 ...

随机推荐

  1. Django模板之自定义过滤器/标签/组件

    自定义步骤: 1.     在settings中的INSTALLED_APPS配置当前app,不然django无法找到自定义的simple_tag. 2.     在app应用中创建templatet ...

  2. Spring 自动装配 byName

    自动装配 byName,这种模式由属性名称(方法名)指定自动装配.Spring 容器看作 beans,在 XML 配置文件中 beans 的 auto-wire 属性设置为 byName.然后,它尝试 ...

  3. 朱刘算法 有向图定根的最小生成树poj3164

    关于为什么不能用Prim求解此类问题,如下 Prim可以看成是维护两个顶点集或者看成维护一颗不断生成的树(感觉前一种说法好一点) 倘若是有向图有三个顶点1.2.3 边的情况如下 1->2:    ...

  4. DFS序--一般都要转化为顶点到每个点

    There is a rooted tree with n nodes, number from 1-n. Root’s number is 1.Each node has a value ai. I ...

  5. 计蒜客 - Fantastic Graph

    题目链接:https://nanti.jisuanke.com/t/31447 知识点: 最大流 题目大意: 给定一个二分图,左边有 $N$ 个点,右边有 $M$ 个点,给出 $K$ 条边.问是否能从 ...

  6. C语言基础知识(一)——关键字

    存储类别说明符变量:auto.register.static.extern._Thread_local.typedef 存储类型限定符:const.volatile.restrict._Atomic ...

  7. C++98/11/17表达式类别

    目标 以下代码能否编译通过,能否按照期望运行?(点击展开) #include <utility> #include <type_traits> namespace cpp98 ...

  8. .NET Core HttpClient源码探究

    前言     在之前的文章我们介绍过HttpClient相关的服务发现,确实HttpClient是目前.NET Core进行Http网络编程的的主要手段.在之前的介绍中也看到了,我们使用了一个很重要的 ...

  9. G1 垃圾回收器简单调优

    G1: Garbage First 低延迟.服务侧分代垃圾回收器. 详细介绍参见:JVM之G1收集器,这里不再赘述. 关于调优目标:延迟.吞吐量 一.延迟,单次的延迟 单次的延迟关系到服务的响应时延, ...

  10. CopyOnWriteArrayList(写入并复制) & CountDownLatch(闭锁)

    ConcurrentHashMap: ①Java 5.0 在 java.util.concurrent 包中提供了多种并发容器类来改进同步容器 的性能.② ConcurrentHashMap 同步容器 ...