环球新消息丨Java动态代理讲解和示例

2023-05-07 20:21:26 来源:清一色财经

Java 动态代理底层原理是基于反射机制实现的,其中最重要的是 InvocationHandler 接口,它定义了一个 invoke() 方法,用于实现对代理类中各个方法的调用,在 invoke() 方法中,可以实现对真实角色的调用,并进行扩展,实现动态代理的效果。


(资料图片)

Java动态代理是一种在运行时创建代理类的机制,动态代理可以在不修改源代码的情况下,在运行时为某个接口动态生成实现类,并且可以拦截接口中的方法调用,从而实现一些特殊的功能。

Java 动态代理底层原理是基于反射机制实现的,其中最重要的是 InvocationHandler 接口,它定义了一个 invoke() 方法,用于实现对代理类中各个方法的调用,在 invoke() 方法中,可以实现对真实角色的调用,并进行扩展,实现动态代理的效果。

动态代理的实现步骤:

1. 创建一个实现 InvocationHandler 接口的类,它必须实现 invoke() 方法;

2. 创建被代理的真实对象;

3. 通过 Proxy 类的 newProxyInstance() 方法创建代理对象,它需要参数:

(1)ClassLoader:类加载器,它是用于加载代理对象字节码的,和被代理对象使用相同的类加载器;

(2)Class[]:字节码数组,它是用于让代理对象和被代理对象有相同方法;

(3)InvocationHandler:它是调用处理器,执行代理对象的方法时,会触发该对象的 invoke() 方法;

4. 通过代理对象调用目标方法,实际上会转到 invoke() 方法中,在 invoke() 方法中,可以进行预处理、调用后处理等工作。

反射上一篇文章讲过了,这里就是反射的一个应用,这个最常见的场景就是给代码加日志,例如在执行某个函数前将请求体加入到日志中,执行后将结果加入到日志中。这样可以在不改变原来代码的基础上来实现。接下来就使用上一篇反射的例子继续扩展。

先写一个Person的接口:

package ReflectTest;public interface PersonInterface {    void printName();    void printAge();}

写他的实现:

package ReflectTest;public class Person implements PersonInterface{    private String name = "xiaoming";    private String age = "12";    @Override    public void printName()    {        System.out.println(name);    }    @Override    public void printAge()    {        System.out.println(age);    }}

然后写动态代理类:

package ReflectTest;import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;public class ProxyHandler implements InvocationHandler {    private Object target;    public ProxyHandler(Object target) {        this.target = target;    }    public Object bind() {        return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(), this);    }    @Override    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {        System.out.println("执行函数前加入日志");        Object result = method.invoke(target, args);        System.out.println("执行函数后加入日志");        return result;    }}

最后测试一下:

package ReflectTest;public class main {    public static void main(String[] args) {        ProxyHandler proxyHandler = new ProxyHandler(new Person());        PersonInterface person = (PersonInterface)proxyHandler.bind();        person.printName();    }}

我们看看结果在执行函数前后执行了其他操作:

SpringAOP其实原理就类似这种。

关键词:

上一篇:今日关注:【致敬劳动者】周卫华:坚守教育情怀 传播中华文化

下一篇:最后一页