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

动态代理

1.什么是动态代理?

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

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

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

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

动态代理和静态代理最大的区别就是:静态代理是编译器确定的代理类,动态代理是运行期确定的代理类。

也就是说:

静态代理其实就是事先写好代理类,可以手动编写也可以使用工具生成,但是它的缺点就是每个业务类都需要一个代理类,因此维护性差,很不灵活也很不方便。

动态代理就是在运行期间,动态的创建目标对象的代理对象。它可以在不修改原始类的情况下

因此,他们两个最终的效果是一样的,都是可以在不修改原始类的代码的情况下,对代理对象的方法进行增强。只不过生成时间不同,和方式不同,静态代理需要给每个类手动创建代理对象,麻烦点罢了。

4.静态代理代码

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

java 复制代码
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.静态代理代码

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

java 复制代码
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类并生成子类,从而实现动态代理的功能。

相关推荐
【D'accumulation】18 分钟前
典型的MVC设计模式:使用JSP和JavaBean相结合的方式来动态生成网页内容典型的MVC设计模式
java·设计模式·mvc
wn53128 分钟前
【Go - 类型断言】
服务器·开发语言·后端·golang
试行33 分钟前
Android实现自定义下拉列表绑定数据
android·java
茜茜西西CeCe38 分钟前
移动技术开发:简单计算器界面
java·gitee·安卓·android-studio·移动技术开发·原生安卓开发
救救孩子把43 分钟前
Java基础之IO流
java·开发语言
小菜yh44 分钟前
关于Redis
java·数据库·spring boot·redis·spring·缓存
宇卿.1 小时前
Java键盘输入语句
java·开发语言
浅念同学1 小时前
算法.图论-并查集上
java·算法·图论
希冀1231 小时前
【操作系统】1.2操作系统的发展与分类
后端
立志成为coding大牛的菜鸟.1 小时前
力扣1143-最长公共子序列(Java详细题解)
java·算法·leetcode