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. React中父子组件数据传递

    Vue.js中父子组件数据传递:Props Down ,  Events Up Angular中父子组件数据传递:Props Down,  Events  Up React中父子组件数据传递:Prop ...

  2. xv6 操作系统的环境搭建

    xv6 是 MIT 设计的一个教学型操纵系统.xv6 可在 Intel X86 框架上运行,为了方便,建议将 xv6 运行在 QEMU 虚拟机器上,本人的实验环境是 ubuntu 18.04 . 1. ...

  3. Java数组声明创建和使用以及多维数组、Arrays类、稀疏数组

    目录 数组概述 数组声明创建 内存分析 java内存分析 堆 栈 方法区 三种初始化 静态初始化 动态初始化 数组的默认初始化 数组的四个基本特点 数组边界 小结: 数组使用 数组基础使用 For E ...

  4. 一言难尽,Jpa这个功能差点让我丢了工作

    故事背景 前阵子,有位朋友在微信上问我数据被删了能不能恢复,我问了下原因,居然是因为一个配置项惹的祸. 故事细节 在 Spring Boot 中使用 jpa 来操作数据库,jpa 就不做详细的介绍了, ...

  5. 利用SQL语句替换织梦DedeCms数据库内容

    找到 然后把命令放在 执行即可,慎重执行 替换文章标题 UPDATE `dede_archives` SET `title`=replace(title,'我我我','你你你') 替换文章正文 UPD ...

  6. 使用for循环疑难问题

    接触js的基本语句之后,有一些疑难杂症在初期很难自己想出来,对我自己来说for输出三角形,倒三角行还有等腰三角形还是有点难度,所以记录一下,以便以后查找 倒三角,需要控制每行的输出个数,此处可以想象为 ...

  7. Vue中计算属性(computed)和监听属性函数watch的比较

    1.计算属性适用的情形我们可能会有这样的需求,一个数据属性在它所依赖的属性发生变化时,也要发生变化,这种情况下,我们最好使用计算属性.例如在下面这个例子中,如果我们使用监听函数,代码就会变得有点冗余. ...

  8. 一元三次方程组求解 luogu P1024

    题目传送门 首先,要明确题目信息,f(x1) * f(x2) < 0, 则一定存在实数根在区间(x1, x2).且所有的根都在[-100, 100)之间.根与根的绝对值之差 >= 1 那么 ...

  9. 分布式事务解决方案Seata

    Seata全称是Simple Extensible Autonomous Transaction Architecture,是由阿里巴巴开源的具有高性能和易用性的分布式事务解决方案. 微服务中的分布式 ...

  10. 高性能可扩展mysql 笔记(四)项目分区表演示

    个人博客网:https://wushaopei.github.io/    (你想要这里多有) 登录日志的分区 如何为Customer_login_log表分区? 从以下两个业务场景入手: 用户每次登 ...