JDK之动态代理(JDK与CGLIB)与静态代理

简介: 代理类在程序运行时创建的代理方式被称为动态代理静态代理中,代理类是自己定义好的,在程序运行之前就已经编译完成。然而动态代理,代理类并不是在Java代码中定义的,而是运行时根据我们在Java代码中的指示动态生成的。相比较静态代理,动态代理的优势在于可以很方便的对代理类的函数进行统一的处理,而不用修改每个代理类中的方法,比如想要在每个代理方法前都加上一个处理方法,静态代理就需要在每个类内部加上这个方法

动态代理


代理类在程序运行时创建的代理方式被称为动态代理

静态代理中,代理类是自己定义好的,在程序运行之前就已经编译完成。然而动态代理,代理类并不是在Java代码中定义的,而是运行时根据我们在Java代码中的指示动态生成的。相比较静态代理,动态代理的优势在于可以很方便的对代理类的函数进行统一的处理,而不用修改每个代理类中的方法,比如想要在每个代理方法前都加上一个处理方法,静态代理就需要在每个类内部加上这个方法


静态代理与动态代理的区别

静态代理

优点

简单、效率高、容易理解

缺点

当目标类增多,代理类也需要增加

当接口方法增加或修改的时候,很多类都需要修改,因为目标类和代理类都实现了相同的接口

动态代理

优点

动态代理中的目标类很多的时候代理类的数量可以很少

修改接口的方法的时候不会影响到代理类

代理类可以自动帮助我们生成无需手动


JDK动态代理


JDK动态代理是基于反射机制,生成一个实现代理接口的匿名类,然后重写方法进行方法增强。在调用具体方法前通过调用InvokeHandler的invoke方法来处理。

他的特点是生成代理类的速度很快,但是运行时调用方法操作会比较慢,因为是基于反射机制的,而且只能针对接口编程,即目标对象要实现接口

public interface Person {
    void sayHi();
}
public class Student implements Person{
    @Override
    public void sayHi() {
        System.out.println("Hi 我是一名学生");
    }
}
public class JDKProxy implements InvocationHandler {
    Object target;
    public Object newProxy(Object obj){
        this.target = obj;
        return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(),this);
    }
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("你好");
        Object invoke = method.invoke(target, args);
        System.out.println("再见");
        return invoke;
    }
}
public class JDKTest {
    public static void main(String[] args) {
        Person proxy = (Person) new JDKProxy().newProxy(new Student());
        proxy.sayHi();
    }
}

a005188a94fc450892ddf3c0d5592a00.png


CGLIB动态代理


Cglib(Code Generation Library)是一个强大的,高性能,高质量的Code生成类库,它是开源的。动态代理是利用 asm 开源包,将目标对象类的 class 文件加载进来,然后修改其字节码生成新的子类来进行扩展处理

    <--引入依赖-->
        <dependency>
            <groupId>cglib</groupId>
            <artifactId>cglib</artifactId>
            <version>3.3.0</version>
        </dependency>
public class Student {
    public void sayHi(){
        System.out.println("Hi 我是一名学生");
    }
}
public class CglibProxy implements MethodInterceptor {
    private Object target;
    public Object newProxy(Object target){
        this.target = target;
        // 增强器
        Enhancer enhancer = new Enhancer();
        enhancer.setCallback(this);
        enhancer.setSuperclass(target.getClass());
        return enhancer.create();
    }
    /**
     *
     * @param o 生成代理类的实例
     * @param method 上文实体类所调用的被代理的方法引用
     * @param objects 方法列表参数
     * @param methodProxy 生成代理类对方法的引用
     * @return
     * @throws Throwable
     */
    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        System.out.println("你好");
        Object invoke = methodProxy.invoke(target, objects);
        System.out.println("再见");
        return invoke;
    }
}
public class CglibTest {
    public static void main(String[] args) {
        CglibProxy proxy = new CglibProxy();
        Student student = (Student) proxy.newProxy(new Student());
        student.sayHi();
    }
}

e07ec90cf73f4847bbe26f4a60429c2e.png


两种代理方式的区别


JDK动态代理是基于反射机制,生成一个实现了接口继承了proxy类的匿名类而Cglib动态代理是基于继承机制,继承被代理类,底层是基于asm第三方框架对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理

JDK动态代理生成类的速度快,后续执行类的方法操作慢而Cglib代理是生成类的速度慢,后续执行类的方法操作快


为什么JDK生成快,执行慢而Cglib与之相反


JDK动态代理通过会拦截方法,通过反射获取模板接口名字、内部方法以及参数,拼接生成一个新的Java代理对象

生成的代理对象不能直接调用被代理对象的方法,而是通过反射所有每次都得用反射调用一次所以执行方法都需要通过反射


Cglib代理实际上通过继承,生成一个子类,就是将代理对象类的class文件加载进来,通过修改其字节码生成子类来处理,生成类的速度慢但是后续执行时候就很快


相关文章
|
5天前
|
监控 Java API
JDK动态代理和CGLIB动态代理
Java动态代理允许在运行时创建代理对象,增强或拦截目标类方法的执行。主要通过两种方式实现:JDK动态代理和CGLIB动态代理。JDK动态代理基于接口,利用`java.lang.reflect.Proxy`类和`InvocationHandler`接口;CGLIB则通过字节码技术生成目标类的子类作为代理,适用于未实现接口的类。两者均用于在方法执行前后添加额外逻辑,如日志记录、权限控制等,广泛应用于AOP框架中。
|
13天前
|
监控 Java API
JDK动态代理和CGLIB动态代理
Java动态代理允许在运行时创建代理对象,增强或拦截目标类的方法调用,无需修改原代码。它有两种主要实现方式:JDK动态代理和CGLIB动态代理。 - **JDK动态代理**:通过`java.lang.reflect.Proxy`类和`InvocationHandler`接口实现,适用于实现了接口的类。它在方法调用前后插入额外逻辑,如日志记录、权限控制等。 - **CGLIB动态代理**:基于字节码技术,为未实现接口的类生成子类作为代理,重写父类方法以添加增强逻辑。适用于没有接口的类,但要求目标类不能是`final`类或方法。
|
1月前
|
Java API 数据安全/隐私保护
探索Java动态代理的奥秘:JDK vs CGLIB
动态代理是一种在 运行时动态生成代理类的技术,无需手动编写代理类代码。它通过拦截目标方法的调用,实现对核心逻辑的 无侵入式增强(如日志、事务、权限控制等)。
54 0
探索Java动态代理的奥秘:JDK vs CGLIB
|
4月前
|
安全 Java 开发者
AOP中的JDK动态代理与CGLIB动态代理:深度解析与实战模拟
【11月更文挑战第21天】面向切面编程(AOP,Aspect-Oriented Programming)是一种编程范式,它通过将横切关注点(cross-cutting concerns)与业务逻辑分离,以提高代码的可维护性和可重用性。在Java开发中,AOP的实现离不开动态代理技术,其中JDK动态代理和CGLIB动态代理是两种常用的方式。本文将从背景、历史、功能点、业务场景、底层逻辑等多个维度,深度解析这两种代理方式的区别,并通过Java示例进行模拟和比较。
208 5
|
6月前
|
Java
安装JDK18没有JRE环境的解决办法
安装JDK18没有JRE环境的解决办法
585 61
|
8天前
|
Oracle Java 关系型数据库
课时4:JDK的安装与配置
课时4:JDK的安装与配置 摘要: 1. JDK安装:从Oracle官网下载适合操作系统的JDK版本,确保关闭防火墙,选择正确的位数(如64位),并进行一键式安装。 2. JDK配置:将JDK的bin目录路径(如D:\Java\jdk1.8.0_74\bin)添加到系统环境变量PATH中,确保Java开发命令(如javac、java)可用。配置完成后,重启命令行工具验证安装是否成功。 通过以上步骤,确保Java开发环境的正确搭建。
|
8天前
|
Java
课时5:JDK安装与配置
课时5:JDK安装与配置,主讲人李兴华。课程详细讲解了JDK的安装步骤和环境配置方法,包括选择安装路径、配置系统环境变量(如path),确保javac和java命令在命令行中可用。建议将所有程序安装在D盘,便于管理。安装完成后,需重启命令行以加载新环境配置,确保Java开发环境正常运行。
|
7月前
|
Oracle Java 关系型数据库
Mac安装JDK1.8
Mac安装JDK1.8
890 4
|
7月前
|
Java 关系型数据库 MySQL
"解锁Java Web传奇之旅:从JDK1.8到Tomcat,再到MariaDB,一场跨越数据库的冒险安装盛宴,挑战你的技术极限!"
【8月更文挑战第19天】在Linux上搭建Java Web应用环境,需安装JDK 1.8、Tomcat及MariaDB。本指南详述了使用apt-get安装OpenJDK 1.8的方法,并验证其版本。接着下载与解压Tomcat至`/usr/local/`目录,并启动服务。最后,通过apt-get安装MariaDB,设置基本安全配置。完成这些步骤后,即可验证各组件的状态,为部署Java Web应用打下基础。
89 1
|
3月前
|
NoSQL 关系型数据库 MySQL
Linux安装jdk、mysql、redis
Linux安装jdk、mysql、redis
260 7