动态代理详解(原理+代码+代码解析)

本文介绍了动态代理的概念、功能,包括AOP编程中的方法拦截、远程方法调用和动态权限校验。对比了动态代理与静态代理的区别,并给出了Java动态代理的实现示例,以及JDK反射机制和CGLIB字节码生成库在动态代理中的作用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

动态代理

在这里插入图片描述

1.什么是动态代理?

动态代理是一种在运行的时候动态的生成代理对象的技术。它在不改变原始类的情况下,对原始类的方法进行拦截或者增强。

2.动态代理可以实现的功能?

使用动态代理可以实现如下常用功能:
1.AOP(面向切面编程):动态代理可以用于实现横切逻辑,例如日志记录,性能监控,事务管理等。通过在方法前后插入代理逻辑,可以实现对目标方法的增强。
2.远程方法调用(RPC):动态代理可以用于远程方法调用的框架,例如使用代理对象作为客户端的远程服务代理,将调用转发给实际的远程服务,并且可以将客户端或服务器的业务逻辑进行增强。
3.动态权限校验:动态代理可以用于动态权限校验,例如在访问某个受限资源的时候(我们假设是个方法),使用这个方法的对应对象的代理对象来调用这个方法的时候,在调用方法前可以增加权限判断的逻辑,如果权限对,就可以进入。

3.动态代理和静态代理的区别?

动态代理和静态代理最大的区别就是:静态代理是编译器确定的代理类,动态代理是运行期确定的代理类。
也就是说:
静态代理其实就是事先写好代理类,可以手动编写也可以使用工具生成,但是它的缺点就是每个业务类都需要一个代理类,因此维护性差,很不灵活也很不方便。
动态代理就是在运行期间,动态的创建目标对象的代理对象。它可以在不修改原始类的情况下
因此,他们两个最终的效果是一样的,都是可以在不修改原始类的代码的情况下,对代理对象的方法进行增强。只不过生成时间不同,和方式不同,静态代理需要给每个类手动创建代理对象,麻烦点罢了。

4.静态代理代码

代码比较简单,不过多解释,在动态代理的代码会详细解释

interface User{
    int age();
    void sayHi(String name);
}
class UserImpl implements User{
    @Override
    public int age() {
        return 18;
    }

    @Override
    public void sayHi(String name) {
        System.out.println("hello"+name);
    }
}
class UserProxy implements User{
    public User user;
    public UserProxy(User user){
        this.user=user;
    }

    @Override
    public int age() {
        return 18;
    }

    @Override
    public void sayHi(String name) {
        System.out.println("before");
        System.out.println("hello:"+name+",你的年龄是:"+user.age());
        System.out.println("after");
    }
}
public class Test10 {
    public static void main(String[] args) {
        User user=new UserImpl();
        UserProxy userProxy=new UserProxy(user);
        userProxy.sayHi("小白");
    }
}

5.静态代理代码

解释在代码后面,请耐心看完

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

interface School{
    void location();
}
class SchoolImpl implements School{

    @Override
    public void location() {
        System.out.println("学校的位置");
    }
}
class SchoolInvocationHandler implements InvocationHandler{
    public Object target;
    public SchoolInvocationHandler(Object target){
        this.target=target;
    }
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("before");
        Object result=method.invoke(target,args);
        System.out.println("after");
        return result;
    }
}
public class Test13 {
    public static void main(String[] args) {
        School school=new SchoolImpl();
        SchoolInvocationHandler handler=new SchoolInvocationHandler(school);
        School schoolProxy=(School) Proxy.newProxyInstance(School.class.getClassLoader(),
                new Class[]{School.class},handler);
        schoolProxy.location();
    }

}

首先,被代理的对象需要实现接口,如下图
在这里插入图片描述
为什么被代理对象要实现接口呢?
因为动态代理要求被代理对象实现接口主要是由Java动态代理机制的设计决定的。在Java中,动态代理是基于接口的,这意味着它使用接口来定义代理对象可以调用的方法集合。
然后,就创建一个调用器,这个所谓的调用器呢,你就可以当他是个工具,当代理对象去调用被代理对象的方法的时候,人家代理对象需要拿着这个调用器去调用方法,要不它调用不了,调用器的代码如下图,这个handler就是调用器
在这里插入图片描述
在这里插入图片描述
然后就创建代理对象,如下图红色圈,下图的绿色圈就是你需要被代理的类,这个类叫啥,你就把这里替换成相应的类名就行了
在这里插入图片描述
之后,当走了下图黄色圈的时候,也就是代理对象调用被代理类的方法的时候
在这里插入图片描述
这时候就相当于代理对象拿着调用器(handler),然后去找代理类,然后调用代理方法,然后走的是invoke,如下图,也就是说,当代理对象开始调用被代理类的方法的时候,就走调用器的invoke的方法
在这里插入图片描述
下面我来解释这三个参数
在这里插入图片描述
Object proxy:就代理对象自己,也就是schoolProxy,
Method method:是代理对象调用的方法,也就是location()方法
Object[] args:相当于location()方法里面的参数,我这个代码没有参数,如果有参数的话,比如location(int age,String name),就把这两个参数放到args这个数组,并且如果是基础数据类型,放到args这个数组就会变成包装类,也就是int变成Integer age
下面来解释下图红圈部分
在这里插入图片描述
method.invoke表示代理对象拿着调用器真正的去调用location方法了
target是被代理的对象
Object result代表的就是location()方法的返回值,如果没有返回值,那么object就是null

6.动态代理的底层

JDK动态代理的底层是通过java的反射机制实现的。
java的反射机制允许程序在运行时动态的获取类的信息(如成员变量,方法,构造函数等),并在运行时调用对象的方法或创建对象,如下源码所示
在这里插入图片描述
但是,动态代理除了可以依靠反射来实现之外,还可以使用字节码生成库来实现,字节码生成库允许我们在允许时生成字节码,从而创建代理对象,例如CGLib,它是一个开源的字节码生成库,它可以在运行时扩展java类并生成子类,从而实现动态代理的功能。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值