代理模式之Java动态代理
1. Overview
Java在java.lang.reflect包下,定义了自己的代理。利用这个包下的类,我们可以在运行时动态地创建一个代理类,实现一个或多个接口。并将方法的调用转发到你所指定的类。因为实际代理是在运行时创建的,所以称为:动态代理。动态代理的原理,就是JVM动态的生成一个实现了指定接口的类,在类中调用了InvocationHandler的invoke方法,实现了权限验证等。
Proxy:完全由java产生的,而且实现了完整的subject接口。
InvocationHandler:Proxy上的任何方法调用都会被传入此类,InvocationHandler控制对RealSubject的访问。
因为Java已经帮助我们创建了Proxy类,我们需要有办法告诉Proxy类你要做什么,我们不能像以前一样把代码写入到Proxy类中,因为Proxy类不是我们实现的。那么我们应该放在哪里?放在InvocationHandler类中,InvocationHandler类是响应代理的任何调用。我们可以吧InvocationHandler想成是代理收到方法调用后,请求做实际工作的对象。
2. java.lang.reflect.Proxy
提供用于创建动态代理类和实例的静态方法,它还是由这些方法创建的所有动态代理类的超类,类的定义及其中方法:
public class Proxy implements java.io.Serializable { //该方法返回指定代理对象所关联的调用处理器 public static InvocationHandler getInvocationHandler(Object proxy); //该方法用于获取指定类装载器和一组接口的动态代理类的类对象 public static Class<?> getProxyClass(ClassLoader loader, Class<?>... interfaces) //判断指定类对象是否是一个动态代理类,当且仅当指定的类通过getProxyClass方法或newProxyInstance方法动态生成为代理类时,返回true。 public static boolean isProxyClass(Class<?> cl); //用于为指定类装载器、一组接口及调用处理器生成动态代理类实例,我们最常用的方法。 public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,InvocationHandler h)}3. java.lang.reflect.InvocationHandler
被代理实例所实现的一个接口,内部只有一个invoke()方法,签名如下;
public Object invoke(Object proxy, Method method, Object[] args)
该方法负责集中处理动态代理类上的所有方法调用。第一个参数既是代理类实例,第二个参数是被调用的方法对象,第三个参数是方法调用的参数值。调用处理器根据这三个参数进行预处理或分派到委托类实例上执行。
每次生成动态代理类对象时都需要指定一个实现了该接口的调用处理器对象,当代理的方法被调用的时候,代理就会把这个调用转发给InvocationHandler,也就会调用它的invoke()方法。(当类使用Proxy产生代理对象时,实际生成了实现了指定接口的类,在这个类中显示的调用了invoke方法,然后调用目标类中的方法,这也是动态代理的原理。)
4. 示例:
情形:自己可以查看修改姓名性别,但是不能修改rate。他人可以查看姓名,性别以及修改rate,但是不能修改姓名性别。
4.1 定义一个接口:
public interface Person {String getName();String getGender();void setName(String name);void setGender(String gender);void setRate(int rate);int getRate();} 4.2 定义实现Person接口类
public class PersonImpl implements Person {String name;String gender;String interests;int rate;public String getName() {return name;}public void setName(String name) {this.name = name;}public String getGender() {return gender;}public void setGender(String gender) {this.gender = gender;}public String getInterests() {return interests;}public void setInterests(String interests) {this.interests = interests;}public int getRate() {return rate;}public void setRate(int rate) {this.rate = rate;}}4.3 定义OwnerInvocationHandler类,表示如果为本人,则可以进行修改查看姓名性别。
public class OwnerInvocationHandler implements InvocationHandler{private Person personBean;public OwnerInvocationHandler(Person personBean){this.personBean = personBean;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args)throws IllegalAccessException {try {if(method.getName().startsWith("get")){//如果方法名为get,就调用person类内的get相应方法return method.invoke(personBean, args);}else if(method.getName().equals("setRate")){ // 如果方法是setRate,则抛出异常throw new IllegalAccessException("access deny");}else if(method.getName().startsWith("set")){ //如果为set,就调用person类内的set相应方法return method.invoke(personBean, args);}else {System.out.println("non method invoke");}} catch (InvocationTargetException e) {e.printStackTrace();}return null; }}4.4 定义NonInvocationHandler类,表示如果不为本人,则可以进行查看姓名性别和修改rate。
public class NonInvocationHandler implements InvocationHandler{//private Person person;public NonInvocationHandler(Person person){this.person = person;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args)throws Throwable {if(method.getName().startsWith("setRate")){return method.invoke(person, args);}else if (method.getName().startsWith("get")){return method.invoke(person, args);} else {System.out.println("non method invoke");return null;}}}4.5 测试类MyDynamicProxy
public class MyDynamicProxy {public Person getOwnerPersonBeanProxy(Person person){return (Person)Proxy.newProxyInstance(person.getClass().getClassLoader(), person.getClass().getInterfaces(), new OwnerInvocationHandler(person));}public Person getNonPersonBeanProxy(Person person){return (Person)Proxy.newProxyInstance(person.getClass().getClassLoader(), person.getClass().getInterfaces(), new NonInvocationHandler(person));}public static void main(String[] args) {MyDynamicProxy mdp = new MyDynamicProxy();mdp.test();}public void test(){ //Person person = getPersonBeanFromDB1();Person personProxy = getOwnerPersonBeanProxy(person);System.out.println(personProxy.getName()); try {personProxy.setRate(2);} catch (Exception e) {System.out.println("can not setRate");} //Person person1 = getPersonBeanFromDB1();Person personProxy2 = getNonPersonBeanProxy(person1);System.out.println(personProxy2.getName());personProxy2.setRate(2);System.out.println(personProxy2.getRate());}private Person getPersonBeanFromDB1(){Person pb = new PersonImpl();pb.setName("remy");pb.setGender("girl");pb.setRate(1);return pb;}}输出结果:
remycan not setRateremy2
1 楼 xyzxingyan 2011-04-06 嗯,学习了,动态代理还是要好好学一下的,谢谢LZ