修炼内功--动态代理
动态代理只能代理接口,不能代理某一个类。
动态代理包含面向切面编程,即在不操作原有代码的基础上可以操作原有代码。
动态代理实现步骤:创建代理类实现java.lang.reflect.InvocationHandler接口;创建目标对象,即需要代理的对象;完成Proxy.newProxyInstance()方法,传入的三个参数是难点我认为,第一个是目标对象的装载器(接口的装载器),第二个参数类,第三个参数为InvocationHandler对象。
写代理类有两种方法,但思想是一致的,先写一种。
public interface HelloWorld{
String say();
}
public class HelloWorldImpl{ public String say(){ System.out.println("hello world in proxyhelloworldimpl!"); return "hello world!"; } } /** *代理类 */ public class DynamicProxy implements InvocationHandler {
private Object target;
public Object newInstance(Object target){ this.target = target; //这里的this代表的是invocationhandler对象,也就是DynamicProxy类 return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getClassInterfaces(),this); } public Object invoke(Object proxy,Method method,Object[] args)throws Throwable{ System.out.println("before proxy"); //target:目标对象 args:传入的参数,比如调目标对象的那个方法 method.invoke(target,args); System.out.println("before proxy"); return null; } }
写一个main类
public ProxyMain{ public static void main(String[] args){ DynamicProxy dp = new DynamicProxy(); HelloWorld hw = new HelloWorldImpl()' HelloWorld hwp =(HelloWorld)dp.newInstance(hw); hwp.say(); } }
//执行结果
before proxy!
hello world in proxyhelloworldimpl!
after proxy!
还有一种写DynamicProxy类的形式是:
public DynamicProxy{
private Object target;
public DynamicProxy(Object target){
this.target = target;
}
public Object invoke(Object proxy,Method method,Object[] args){
System.out.println("before proxy");
method.invoke(target,args);
System.out.println("after proxy");
}
}
/**
*都是利用Proxy.newProxyInstance方法完成代理对象
*/
public ProxyMain{
public void static main(String[] args){
HelloWorld hw = new HelloWorldImpl();
InvocationHandler dp = new DynamicProxy(hw);
HelloWorld hwi = (HelloWorld)Proxy.newProxyInstance(HelloWorld.class.getClassLoader,new Class<?>[]{HelloWorld.class},dp);
hwi.say();
}
}
执行结果是和上一个代理类一样的结果。
动态代理的优点就是利用InvocationHandler实现类可以代理接口,不用重写代理类,节省了代码量,达到复用的目的。
//外面的东西容易忘记,内功才是入骨的东西。