一,动态代理
我发现Spring框架中的动态代理是一种非常强大的机制,它可以在运行时为接口或类创建动态代理,然后通过这些代理在方法调用前后添加额外的行为。在后续Spring的AOP(面向切面编程)支持中扮演了关键角色。
二,应用目的
Spring动态代理主要用于以下目的:
- 事务管理:在方法执行前后自动开始和提交事务。
- 安全性:在方法执行前进行权限检查。
- 性能监控:在方法执行前后添加日志来监测性能
三,核心组件
- Proxy:代理是动态创建的类,它在目标类的基础上增加了额外的处理逻辑。当通过代理对象调用方法时,代理会先调用与之相关的拦截器链。
- InvocationHandler(JDK动态代理):这是实际的拦截逻辑所在,定义了在调用目标方法前后应当执行的操作。每个代理实例都有一个关联的调用处理程序。 当在代理实例上调用方法时,方法调用将被编码并分派到其调用处理程序的invoke方法。(接口中仅有Invoke方法)
- invoke:处理代理实例上的方法调用并返回结果。 当在与之关联的代理实例上调用方法时,将在调用处理程序中调用此方法。
四, invoke(object proxy, method,Object[] args)方法参数
- proxy ---- 调用该方法的代理实例
- method----所述方法对应于调用代理实例上的接口方法的实例。 方法对象的声明类将是该方法声明的接口,它可以是代理类继承该方法的代理接口的超级接口。
- args-----包含的方法调用传递代理实例的参数值的对象的阵列,或null如果接口方法没有参数。
五,例(案例参考租房需求代理)
1.创建动态代理
// 租赁公司用这个类,自动生成代理类!
public class ProxyInvocationHandler implements InvocationHandler {// 被代理的接口private Rent rent;public void setRent(Rent rent) {this.rent = rent;}// 生成代理类public Object getProxy() {return Proxy.newProxyInstance(this.getClass().getClassLoader(), rent.getClass().getInterfaces(), this);}// 代理类要执行的,方法增强public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {// 调用代理对象,就是使用真实对象调用!Object result = method.invoke(rent, args);return result;}
}
这段代码定义了一个ProxyInvocationHandler类,实现了Java的InvocationHandler接口,用于创建一个动态代理。这个类中有几个关键部分:
- 私有成员变量
rent
:它是Rent接口的一个实现,代表被代理的对象。 - setRent 方法:用于设置代理的目标对象。
- getProxy 方法:利用Proxy
.
newProxyInstance方法生成代理对象。这个方法需要三个参数:类加载器,接口数组(代理类需要实现的接口),以及一个InvocationHandler实例(当前对象this)。 - invoke 方法:这是代理实例的方法调用时实际执行的代码。invoke方法可以在调用目标方法前后添加自定义逻辑,本例中只是简单地调用了目标对象的相应方法。
2.定义租房接口
//租房
public interface Rent {public void rent();
}
3.定义房东租房任务
//模拟房东出租房屋
public class Host implements Rent{public void rent(){system.out.println("房东要出租房屋")}
}
4.实现动态代理
package com.kuang.demo03;public class Client {public static void main(String[] args) {// 生成被代理对象Host host = new Host();// 代理角色:现在没有ProxyInvocationHandler pih = new ProxyInvocationHandler();// 通过调用程序处理角色来处理我们要调用的接口对象---也就是可以调用host里面的任务pih.setRent(host);Rent proxy = (Rent) pih.getProxy();proxy.rent();}
}
代码定义了一个客户端类Client,演示如何使用动态代理。这里的重点在于如何创建代理对象并通过它来调用方法:
- Host实例化:首先创建了一个Host对象,Host是Rent接口的实现类,代表租赁服务的具体实现。
- ProxyInvocationHandler实例化:接着创建一个ProxyInvocationHandler的实例。这个类是用来定义代理逻辑的处理器。
- 设置代理的目标对象:通过pih.setRent(host)将host对象设置为代理的目标对象。
- 获取代理对象:通过pih.getProxy()
;
获取代理对象。这一步是通过代理处理器和Java的Proxy类动态生成一个符合Rent接口的代理对象。 - 通过代理对象调用方法:最后通过代理对象调用Rent()方法。如果ProxyInvocationHandler的invoke方法可以添加特定逻辑(如打印日志、权限校验等),它们会在这一步执行。