专业的JAVA编程教程与资源

网站首页 > java教程 正文

Java动态代理讲解和示例(java动态代理实现三种方式)

temp10 2024-10-04 12:33:03 java教程 9 ℃ 0 评论

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

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

Java动态代理讲解和示例(java动态代理实现三种方式)

动态代理的实现步骤:

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其实原理就类似这种。

本文暂时没有评论,来添加一个吧(●'◡'●)

欢迎 发表评论:

最近发表
标签列表