java基础知识(一) JDK动态代理
2014-02-24 23:41
351 查看
一、简单说明
使用java动态代理可以使得面向切面编程变得容易,也可以轻松屏蔽掉一些被代理对象的实现细节。中JDK态代理只能代理一个或者多个接口,因此使用此种方法被代理的类对象一定是实现了一个接口的。
二、相关的类和方法
java.lang.reflect.ProxyProxy
提供用于创建动态代理类和实例的静态方法:
Static Object newProxyInstan(ClassLoaderloader,
Class[] interfaces, InvocationHandler h)
该方法返回代理类的一个实例,返回后的代理类可以当作被代理类使用,它与代理对象关联,当请求分发到代理对象后,会自动执行h.invoke()方法, invoke方法就是我们用来做N多事情的地方其中loader和interfaces的获取方式有两种,以loader为例,一种是对象.getClass().getClassLoader(),一种是类.class.getClassLoader()。
java.lang.reflect.InvocationHandlerInvocationHandler
是代理实例的调用处理程序实现的接口。处理程序在实现InvocationHandler接口时只需要实现一个方法:
Object invoke(Object proxy, Method method, Object[] args)
其中的proxy是代理对象,method是被代理对象的方法,后面会调用method.invoke(target,args),args是方法的参数列表。
三、具体应用
动态代理的原理图如下所示:
下面通过新建一个工程来说明使用上述方法可以实现动态代理多个实现了接口的类,且代理类扩展了委托类的多个方法,在原方法的基础上,将日志输出到原始的输出结果前后。建立的工程如图所示:
其中各个文件的源代码如下:
输出结果如下:
四、经验教训
中间出错的地方是在DaoHandler的构造函数处,将
this.target = target
写成了
target = this.target,
从而使得target的值为null,抛出了NUllPointerException异常。
使用java动态代理可以使得面向切面编程变得容易,也可以轻松屏蔽掉一些被代理对象的实现细节。中JDK态代理只能代理一个或者多个接口,因此使用此种方法被代理的类对象一定是实现了一个接口的。
二、相关的类和方法
java.lang.reflect.ProxyProxy
提供用于创建动态代理类和实例的静态方法:
Static Object newProxyInstan(ClassLoaderloader,
Class[] interfaces, InvocationHandler h)
该方法返回代理类的一个实例,返回后的代理类可以当作被代理类使用,它与代理对象关联,当请求分发到代理对象后,会自动执行h.invoke()方法, invoke方法就是我们用来做N多事情的地方其中loader和interfaces的获取方式有两种,以loader为例,一种是对象.getClass().getClassLoader(),一种是类.class.getClassLoader()。
java.lang.reflect.InvocationHandlerInvocationHandler
是代理实例的调用处理程序实现的接口。处理程序在实现InvocationHandler接口时只需要实现一个方法:
Object invoke(Object proxy, Method method, Object[] args)
其中的proxy是代理对象,method是被代理对象的方法,后面会调用method.invoke(target,args),args是方法的参数列表。
三、具体应用
动态代理的原理图如下所示:
下面通过新建一个工程来说明使用上述方法可以实现动态代理多个实现了接口的类,且代理类扩展了委托类的多个方法,在原方法的基础上,将日志输出到原始的输出结果前后。建立的工程如图所示:
其中各个文件的源代码如下:
package c 4000 om.myproxy.test; public interface UserDaoImpl { public Boolean findUserByName(String name); }
package com.myproxy.test; public class UserDao implements UserDaoImpl{ @Override public Boolean findUserByName(String name) { // TODO Auto-generated method stub if(name.equals("张无忌")) { System.out.println("查找姓名为" + name + "的用户信息成功!"); return true; } else { System.out.println("查找姓名为" + name + "的用户信息失败!"); return false; } } }
package com.myproxy.test; public interface DocumentDaoImpl{ public Boolean findDocumentById(int id); }
package com.myproxy.test; public class DocumentDao implements DocumentDaoImpl{ @Override public Boolean findDocumentById(int id) { // TODO Auto-generated method stub if(id == 1001) { System.out.println("查找id为" + id + "的文档信息成功!"); return true; } else { System.out.println("查找id为" + id + "的文档信息失败!"); return false; } } }
package com.myproxy.test; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.util.Calendar; import java.util.GregorianCalendar; public class DaoHandler implements InvocationHandler{ private Calendar calendar; private Object target; public DaoHandler(Object target){ this.target = target; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { // TODO Auto-generated method stub Object result = null; beforeInvoke(); result = method.invoke(target, args); afterInvoke(); return result; } public void beforeInvoke() { calendar = new GregorianCalendar(); int hour = calendar.get(Calendar.HOUR_OF_DAY); int minute = calendar.get(Calendar.MINUTE); int second = calendar.get(Calendar.SECOND); String time = hour + ":" + minute + ":" + second; System.out.println("调用时间:" + time); } public void afterInvoke() { System.out.println("方法调用结束!" ); } }
package com.test; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Proxy; import com.myproxy.test.*; public class ProxyTest { public static void main(String[] args) { // TODO Auto-generated method stub DaoHandler handler = null; UserDao user = new UserDao(); handler = new DaoHandler(user); UserDaoImpl userProxy = (UserDaoImpl)Proxy.newProxyInstance (user.getClass().getClassLoader(), user.getClass().getInterfaces(), handler); userProxy.findUserByName("张无忌"); userProxy.findUserByName("令狐冲"); System.out.println("-----------------"); DocumentDaoImpl doc = new DocumentDao(); handler = new DaoHandler(doc); DocumentDaoImpl docProxy = (DocumentDaoImpl)Proxy.newProxyInstance (doc.getClass().getClassLoader(), doc.getClass().getInterfaces(), handler); docProxy.findDocumentById(1001); docProxy.findDocumentById(1002); } }
输出结果如下:
四、经验教训
中间出错的地方是在DaoHandler的构造函数处,将
this.target = target
写成了
target = this.target,
从而使得target的值为null,抛出了NUllPointerException异常。
相关文章推荐
- [Java]Spring AOP基础知识-动态代理
- JAVA基础知识之JVM-——动态代理(AOP)
- 学习Spring必学的Java基础知识(2)----动态代理
- 《精通Spring4.X企业应用开发实战》读后感第七章(AOP基础知识、jdk动态代理,CGLib动态代理)
- Java基础:JDK动态代理源码分析
- Java基础-jdk动态代理与cglib动态代理区别
- 学习Spring必学的Java基础知识(2)----动态代理
- Java基础加强:细说JDK动态代理的实现原理
- 第一章 Java基础之JDK动态代理
- Java基础-JDK动态代理
- Java基础学习总结(109)——Jdk动态代理和cglib动态代理总结
- java基础知识学习之代理深入学习(用动态代理实现AOP,FacotryBean)
- 【Java基础】——JDK动态代理VSCGLIB动态代理
- java 框架基础知识(2)----动态代理-->Spring AOP
- java动态代理(JDK和cglib)
- java动态代理(JDK和cglib)
- Java JDK动态代理实现自己的事务管理器
- Java基础加强---动态代理
- Java探索之——动态代理(JDK和CGlib方式)
- Java动态代理机制详解(JDK 和CGLIB,Javassist,ASM)