文章目录
- 1. 相关概念
- 2. 步骤
- 3. 举例
问题:
假设一个java项目有100个java类,每个java有10个方法,这总共有1000个方法,现在有这样一个需求,需要在每个java方法加上2句话:在方法执行前输出这个方法开始执行;在方法执行后输出这个方法已经完成。这么多方法不可能在每个方法中一一实现
解决办法:动态代理
1. 相关概念
Proxy :专门完成代理的操作类,是所有动态代理类的父类。通过此类为一个或多个接口动态地生成实现类。
创建一个动态代理类所对应的 Class 对象:
static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,InvocationHandler h)
:直接创建一个动态代理对象
2. 步骤
- 创建一个实现接口 InvocationHandler 的类,它必须实现 invoke方法,以完成代理的具体操作。
- 创建被代理的类以及接口
- 通过 Proxy 的静态方法
newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h)
创建一个 Subject 接口代理 - 通过 Subject 代理调用 RealSubject 实现类的方法
3. 举例
接口ITestDemo
package day14;public interface ITestDemo {void test1();void test2();
}
类TestDemoImpl实现接口ITestDemo
package day14;public class TestDemoImpl implements ITestDemo {@Overridepublic void test1() {System.out.println("执行test1()方法");}@Overridepublic void test2() {System.out.println("执行test2()方法");}}
动态代理类ProxyDemo
package day14;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;/*** 动态代理类* @author 14532**/
public class ProxyDemo implements InvocationHandler{Object obj;//被代理的对象public ProxyDemo(Object obj) {this.obj = obj;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println(method.getName() + "方法开始执行");Object result = method.invoke(this.obj, args);//执行的是指定代理对象的指定方法System.out.println(method.getName() + "方法执行完毕");return null;}
}
Test.java:
package day14;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;public class Test2 {public static void main(String[] args) {ITestDemo test = new TestDemoImpl();/*** 注意:如果一个对象想要通过Proxy.newProxyInstance方法被代理* 那么这个对象的类一定要有相应的接口* 就像本类中的ITestDemo接口和实现类TestDemoImpl*/test.test1();test.test2();System.out.println("=============下面执行动态代理================");/*** 需求:* 在执行test1和test2方法时需要加入一些东西* 在执行方法前打印test1或test2开始执行* 在执行方法后打印test1或test2执行完毕* 打印的方法名要和当时调用方法保持一致*/InvocationHandler handler = new ProxyDemo(test);/*** Proxy.newProxyInstance(ClassLoader, interface, h)* 参数1是代理对象的类加载器* 参数2是被代理的对象的接口* 参数3是代理对象* * 返回的值就是成功被代理后的对象,返回的是Object类型,需要根据当时情况去转换类型*/ITestDemo t= (ITestDemo)Proxy.newProxyInstance(handler.getClass().getClassLoader(), test.getClass().getInterfaces(), handler);t.test1();System.out.println("----------------------------------");t.test2();}
}
运行结果:
注意:如果一个对象想要通过Proxy.newProxyInstance方法被代理,那么这个对象的类一定要有相应的接口,就像本类中的ITestDemo接口和实现类TestDemoImpl