需要了解两个类 Proxy:代理,invocationHandler:调用代理程序
在此之前,我们要思考class和interface的区别和关系:
所有的interface类型的变量都是通过向上转型并指向某个实例的
CharSequence charSequence=new StringBuffer();
有没有可能不编写实现类,自动生成接口所指向的实例呢?
答案是可以的,Java标准库提供了对应的机制:可以在运行期间动态的创建某个interface的实例,什么叫做动态的创建呢?我们先看一下静态情况下是怎样的。
1.编写接口
package com.dreamcold.aop.demo02;
public interface Hello {
void sayHello();
}
package com.dreamcold.aop.demo02;
public class HelloWorld implements Hello {
public void sayHello(){
System.out.println("Hello World");
}
}
3.测试
package com.dreamcold.aop.demo02;
public class Demo01 {
public static void main(String[] args) {
Hello hello=new HelloWorld();
hello.sayHello();
}
}
还有另外一种方式是动态代码,过程如下:
1.先定义接口
package com.dreamcold.aop.demo03;
public interface Hello {
void sayGoodMorning(String name);
}
2.实现动态代理
package com.dreamcold.aop.demo03;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class HelloWorld {
public static void main(String[] args) {
//invocationHandler负责接口的方法调用
InvocationHandler handler=new InvocationHandler() {
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//打印方法
System.out.println(method);
if(method.getName().equals("sayGoodMorning")){
System.out.println("Good Morning "+args[0]);
}
return null;
}
};
//负责创建interface实例
Hello hello=(Hello) Proxy.newProxyInstance(
Hello.class.getClassLoader(),
new Class[]{Hello.class},
handler
);
hello.sayGoodMorning("dreamcold");
}
}
在运行的时候创建一个interface实例的方法如下:
实质:动态代理实际上就是JDK在运行期间创建class字节码并加载的过程,它并没有什么黑魔法,把上面的静态实现类:
public class HelloDynamicProxy implements Hello {
InvocationHandler handler;
public HelloDynamicProxy(InvocationHandler handler) {
this.handler = handler;
}
public void morning(String name) {
handler.invoke(
this,
Hello.class.getMethod("morning", String.class),
new Object[] { name });
}
}
其实就是JDK帮助我们自动编写了一个上述类(不需要源代码,可以直接生成字节码),并不存在直接可以直接实例化接口的黑魔法。
学习自链接:狂神说
原文:https://www.cnblogs.com/mengxiaoleng/p/14940385.html