public class Main {
public static void main(String[] args) {
// System.out.println("Hello World!");
// WaitWrap waitWrap = new WaitWrap(new Waiterimpl());
// waitWrap.start();
//装饰者有个弊端 对其他方法不增强的 ,都要调用一下
/**
* 通过Proxy的newProxyInstance方法来创建我们的代理对象,我们来看看其三个参数
* 第一个参数 handler.getClass().getClassLoader() ,我们这里使用handler这个类的ClassLoader对象来加载我们的代理对象
* 第二个参数realSubject.getClass().getInterfaces(),我们这里为代理对象提供的接口是真实对象所实行的接口,表示我要代理的是该真实对象,这样我就能调用这组接口中的方法了
* 第三个参数handler,我们这里将这个代理对象关联到了上方的 InvocationHandler 这个对象上
*/
Waiter waiterimpl = (Waiter) Proxy.newProxyInstance(Main.class.getClassLoader(), Waiterimpl.class.getInterfaces(), new InvocationHandler() {
//args 是方法中的参数
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if ("stop".equalsIgnoreCase(method.getName())) {//对stop方法增强
System.out.println("检查添天气是否良好");
method.invoke(new Waiterimpl(), args);
System.out.println("检查添路口是否畅通");
} else {
method.invoke(new Waiterimpl(), args);
}
return null;
}
});
waiterimpl.start();
waiterimpl.stop(1, 2);
waiterimpl.destroy();
}
}
生成的class字节码
同时我们一定要记住,通过 Proxy.newProxyInstance 创建的代理对象是在jvm运行时动态生成的一个对象,它并不是我们的InvocationHandler类型,也不是我们定义的那组接口的类型,而是在运行是动态生成的一个对象,并且命名方式都是这样的形式,以$开头,proxy为中,最后一个数字表示对象的标号。