AOP之动态代理

什么是AOP

在上一章的学习中,我们知道Spring一直致力于简化我们的Java开发,并且用到了依赖注入(Dependency Injection)与AOP(Aspect-Oriented Programming)这两项非常重要的技术:

  • DI主要解决了在类和类之间有依赖关系的时候,如何通过注入的方式(属性注入、构造器注入)形成松耦合

  • 而今天要学习的AOP则是考虑如何把散落在应用中多处相同的功能剥离出来,使得这些剥离出来的逻辑与业务逻辑相分离的问题。

让我们先来看一个生活中的案例:

每家每户都有一个电表来监控用电量,这样电力公司就知道应该收取多少费用了。虽然每一台用电设备都可以自装一个用电统计的硬件,但这样是极不合理,成本上也不划算的,因为每个用电设备更多关注的是自身的功能是否完善的问题,吸尘器考虑的是清洁效果,微波炉考虑的是加热效果等等。电力公司只需要在合适的地方,比如用电线路入户的地方,统一安装一个电表,就能很好的解决监控所有用电设备电量的问题。

软件系统中的一些功能就像我们家里的电表一样。这些功能需要用到应用程序的多个地方,但是我们又不想在每个点都明确调用它们。

在软件开发中,散布于应用中多处的功能,被称为横切关注点(cross-cutting concern)。通常来讲,这些横切关注点从概念上是与应用的业务逻辑相分离的(但往往会直接嵌入到应用的业务逻辑中)。把这些横切关注点与业务逻辑相分离正是AOP所要解决的问题。

代理模式

  1. 代理模式的概念

    AOP在实现上采用了设计模式中的动态代理模式,因此,在深入学习SpringAOP之前,我们先来一起了解和学习一下这种强大的设计模式。

    代理模式的定义:为其他对象提供一种代理,以控制对这个对象的访问。换句通俗的话来说,它是一种使用代理对象来执行目标对象的方法,并在代理对象中增强目标对象方法的一种设计模式。

    生活中最常见的代理模式就是”中介“。假如说我现在想买一辆二手车,虽然我可以自己去找车源,做质量检测等一系列的车辆过户流程,但是这确实太浪费我得时间和精力了。我只是想买一辆车而已为什么我还要额外做这么多事呢?于是我就通过中介公司来买车,他们来给我找车源,帮我办理车辆过户流程,我只是负责选择自己喜欢的车,然后付钱就可以了。

代理模式的功能主要是起到增强方法和权限拦截的作用。

  1. 代理模式的好处

    从上图我们可以看出,在程序设计中使用代理模式的一些好处:

    • 中介隔离:在调用方(Caller)不能或不想直接与目标对象(Target)打交道的时候,代理对象(Proxy)可以起到两者之间中介的作用。
    • 开闭原则:我们可以通过给代理对象增加新的功能来扩展目标对象的功能,这样我们只需要修改代理类,而不需要修改目标类,符合代码设计的OCP原则(Open Closed Principle,对扩展是开放的,对修改是关闭的)。
  2. 代理模式的种类

    根据代理对象创建的不同,分为两种代理模式:

    • 静态代理:由程序员或者特定工具生成源代码来产生代理对象。在程序运行前,代理类的字节码文件(.class)就已经存在了
    • 动态代理:在程序运行期间,运用反射机制、字节码生成技术来产生代理类和实例。

静态代理模式

Proxy—代理

​ 要实现静态代理,我们首先使用接口的方式来封装被代理的行为:

​ 接口:

public interface IUserDao {
    void save();
}

分别让目标和代理都来实现这个接口,这样,对于调用方来说,无论和代理还是目标打交道,执行的代码都是一致的,都可以执行相同的行为。

public class UserDaoImpl implements IUserDao {
    @Override
    public void save() {
        System.out.println("用户数据保存!");
    }
}

为目标方编写对应的代理类,在其中增加新的服务功能:

package com.tuling.dao;
public class UserDaoProxy implements IUserDao{

    private IUserDao userDao;

    public UserDaoProxy(IUserDao userDao) {
        this.userDao = userDao;
    }
    @Override
    public void save() {
        System.out.println("开启事务...");
        userDao.save();//执行目标对象方法
        System.out.println("提交事务...");
    }
}

从上面可以看到,我们最终让客户执行了购买行为,除此之外,为了让客户享受更为完善的服务,我们还扩展了寻找车源、质量检测、售后咨询等其它服务。在一个复杂的应用中,我们当然不是仅仅打印几行字,我们可以封装单独的方法来做这些事情,甚至还可以调用其它的类来执行这些辅助逻辑。

测试代码:

package com.tuling.dao;
public class MainTest {
    public static void main(String[] args) {
        //目标对象
        UserDaoImpl userDao = new UserDaoImpl();
        //代理对象
        UserDaoProxy proxy = new UserDaoProxy(userDao);
        //执行代理对象的方法
        proxy.save();
    }

    /*
 * 静态代理总结:
 * 1、可以做到在不修改目标对象的功能前提下,增强目标对象。
 * 2、但是由于要和目标对象实现相同的接口,如果目标方法过多,或
者
 * 目标对象一旦新增方法,代理对象都要一并维护。
 */
}

动态代理模式

静态代理模式最大的缺陷就是,我们需要为每一个被代理的目标类都编写一个代理类。而动态代理可以很好的解决这个问题。JDK的Proxy和开源框架CGLIB可以分别在不同的情况帮助我们生成代理。

  • JDK动态代理

当目标的被代理方法抽取了接口时,可以使用JDK Proxy。

接口:

package com.tuling.proxy;
/**
* 接口
* @author fred
*
*/
public interface IUserDao {
    void save();
}

目标类:

package com.tuling.proxy;
public class UserDaoImpl implements IUserDao {
    @Override
    public void save() {
        System.out.println("用户数据保存!");
    }
}

测试类:

package com.tuling.proxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class MainTest {
    public static void main(String[] args) {
        //目标对象
        UserDaoImpl userDao = new UserDaoImpl();
        //代理对象
        IUserDao proxy = (IUserDao) Proxy.newProxyInstance(
            userDao.getClass().getClassLoader(), //目标对
            象的类加载器
            userDao.getClass().getInterfaces(), //目标对
            象的接口数组
            new InvocationHandler() {//事件处理,执行目标对
                象的方法时,会触invoke方法

                    @Override
                    public Object invoke(Object proxy, Method
                                         method, Object[] args) throws Throwable {
                    System.out.println("开启事务...");
                    //执行目标对象方法
                    Object obj = method.invoke(userDao,
                                               args);
                    System.out.println("提交事务...");
                    return obj;
                }
            });
        //调用代理对象方法
        proxy.save();

        /*
 * JDK动态代理总结:
 * 1、目标对象必须要有接口,否则不能实现动态代理。
 * 2、代理对象必须强转为接口类型。
 */
    }
}

invoke方法会在代理对象的被代理方法调用的时候触发。

CGLIB

当目标类没有实现接口时,我们可以通过开源的CGLIB来实现。

使用maven引入cglib库:

<dependency>
    <groupId>cglib</groupId>
    <artifactId>cglib</artifactId>
    <version>3.3.0</version>
</dependency>

也可以引入spring-core库,该库中包含了CGLIB。

<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-core</artifactId>
    <version>5.1.2.RELEASE</version>
</dependency>

目标类:

package com.tuling.cglib;
/**
* 目标对象,没有任何接口
* @author fred
*
*/
public class UserDaoImpl{
    public void save() {
        System.out.println("用户数据保存!");
    }
}

代理工厂类:

package com.tuling.cglib;
import java.lang.invoke.MethodHandleInfo;
import java.lang.reflect.Method;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
public class ProxyFactory implements MethodInterceptor{

    //目标对象
    private Object target;
    public ProxyFactory(Object target) {
        super();
        this.target = target;
    }

    //给目标对象创建一个代理对象
    public Object getProxyInstance(){
        //1.增强器
        Enhancer en = new Enhancer();
        //2.设置目标对象的类加载器
        en.setClassLoader(target.getClass().getClassLoader());
        //3.设置这个动态代理类的父类
        en.setSuperclass(target.getClass());
        //4.设置要传入的拦截器
        en.setCallback(this);
        //5.创建子类(代理对象)
        return en.create();
    }
    @Override
    public Object intercept(Object arg0, Method method,
                            Object[] args, MethodProxy arg3) throws Throwable {
        System.out.println("开启事务...");
        //执行目标对象的方法
        Object obj = method.invoke(target, args);
        System.out.println("提交事务...");
        return obj;
    }
}

测试类:

package com.tuling.cglib;
public class MainTest {
    public static void main(String[] args) {
        //目标对象
        UserDaoImpl userDao = new UserDaoImpl();
        //代理对象
        UserDaoImpl proxy =
            (UserDaoImpl) new ProxyFactory(userDao).getProxyInstance();
        //执行代理对象方法
        proxy.save();
    }
}

总结

  • 静态代理需要自己手动编写代理类和目标方法。

  • 动态代理就不需要自己手动实现代理类和目标方法,但动态代理的目标类要必须实现接口!

  • Cglib 代理的目标类就不需要实现接口!但目标类不能被final修饰!

Spring AOP 编程的实现原理就是 动态代理和Cglib 代理,当目标类实现接口时使用动态代理,没有则Cglib代理。

  • 3
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring AOP动态代理是利用JDK的反射机制,在程序执行的时候动态创建代理类对象,并动态指定要代理的对象。它的作用是在不改变已有代码的基础上,通过代理来横向拓展增强其他的功能,减少重复的代码,专注业务逻辑代码,解耦业务代码与非业务代码的关系。\[1\] 在Spring AOP中,实现动态代理的方式有两种:静态代理和基于配置的动态代理。静态代理是通过手动编写代理类来实现,而基于配置的动态代理是通过在配置文件中进行配置来建立代理对象与目标对象之间的联系。\[2\] 在配置文件中,可以使用ProxyFactoryBean来创建代理对象,并指定要代理的目标对象、拦截器和代理接口。通过这种方式,可以实现对目标对象的代理,并在代理对象中添加额外的功能。\[2\] 然而,在配置过程中可能会遇到一些问题,比如依赖注入失败的异常。这可能是因为没有找到符合要求的bean,或者注入的依赖项没有正确配置。需要检查配置文件和注解的使用,确保所有的依赖项都正确配置和注入。\[3\] #### 引用[.reference_title] - *1* [SpringAOP动态代理](https://blog.csdn.net/a55566999/article/details/125428712)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v4^insert_chatgpt"}} ] [.reference_item] - *2* *3* [Spring AOP动态代理方式](https://blog.csdn.net/m0_53536589/article/details/123151359)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v4^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值