动态代理

 从JDK1.3开始,Java就引入了动态代理的概念。动态代理(Dynamic Proxy)可以帮助你减少代码行数,真正提高代码的可复用度。例如,你不必为所有的类的方法里面都写上相同的Log代码行,取而代之的是实用类的动态代理类。当然,这种便利是有条件的。本文简单介绍Java动态代理的原理,并实现一个被代理的Servlet创建,和调用的过程。
 
1.代理模式(Proxy Pattern)
在JDK1.3以前,代理模式就已流行,所以得代理模式是生成一个和类相同接口的代理类,用户通过使用代理类来封装某个实现类。如图1,其目的是加强实现类的某个方法的功能,而不必改变原有的源代码。

2.动态代理(Dynamic Proxy)
随着Proxy的流行,Sun把它纳入到JDK1.3实现了Java的动态代理。动态代理和普通的代理模式的区别,就是动态代理中的代理类是由java.lang.reflect.Proxy类在运行期时根据接口定义,采用Java反射功能动态生成的。和java.lang.reflect.InvocationHandler结合,可以加强现有类的方法实现。如图2,图中的自定义Handler实现InvocationHandler接口,自定义Handler实例化时,将实现类传入自定义Handler对象。自定义Handler需要实现invoke方法,该方法可以使用Java反射调用实现类的实现的方法,同时当然可以实现其他功能,例如在调用实现类方法前后加入Log。而Proxy类根据Handler和需要代理的接口动态生成一个接口实现类的对象。当用户调用这个动态生成的实现类时,实际上是调用了自定义Handler的invoke方法。
  

3.动态代理Servlet
              虽然Web Application Server的产品很多,但Servlet的处理原理是相似的:动态加载Servlet,调用Servlet的init方法(只被调用一次),并保存到Servlet容器;Servlet使用时,调用Servlet的service方法。本文动态代理Servlet接口,使其init和service被调用时会在控制台打出方法调用前后信息。
首先实现2个Servlet,DefaultServlet和UserServlet
 
package org.colimas.servlet;
 
import javax.servlet.Servlet;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
 
public class DefaultServlet extends HttpServlet implements Servlet {
              public void init() throws ServletException {
                            super.init();
                            System.out.println(DefaultServlet.class.getName()+":Running init");
              }
 
              public String getServletInfo() {
                            return DefaultServlet.class.getName();
              }
}
 
package org.colimas.servlet;
 
import java.io.IOException;
 
import javax.servlet.Servlet;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
 
 
public class UserServlet extends HttpServlet implements Servlet {
 
              private static final long serialVersionUID = -7016554795165038652L;
             
              public void init() throws ServletException {
                            super.init();
                            System.out.println(UserServlet.class.getName()+":Running init");
              }
              protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
                                         
                            System.out.println(UserServlet.class.getName()+":Do UserSErvlet Get");
              }
              public String getServletInfo() {
                            return UserServlet.class.getName();
              }            
             
 
}
 
然后实现InvocationHandler
package org.colimas.webapp;
 
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
 
import javax.servlet.Servlet;
 
public class ServletHandler implements InvocationHandler {
 
              private Servlet obj;
             
              public ServletHandler(Servlet obj){
                            this.obj=obj;
              }
              public Object invoke(Object arg0, Method arg1, Object[] arg2)
                                          throws Throwable {
                           
                            if(arg1.getName().compareTo("init")==0) //调用init时
                            {
                                          System.out.println(obj.getServletInfo()+":Init servlet starting..."); //增加控制台输出。
                                          arg1.invoke(obj,arg2); //调用init方法
                                          System.out.println(obj.getServletInfo()+":Init servlet ending..."); //增加控制台输出。
                            }else if(arg1.getName().compareTo("service")==0){ //调用service时
                                          System.out.println(obj.getServletInfo()+":service starting..."); //增加控制台输出。
 
                                          arg1.invoke(obj,arg2); //调用service方法。
                                          System.out.println(obj.getServletInfo()+":service ending..."); //增加控制台输出。
                           
                            }
                            return null;
              }
 
}
 
实现Servlet的调用
package org.colimas.webapp;
 
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;
 
import javax.servlet.Servlet;
import javax.servlet.ServletConfig;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
 
public class ServletWrapperImp {
   
    private Class servletClass;
    private ServletConfig config;
    private String _servletname;
    private Servlet _theServlet;
              private ServletContext context;
    public ServletWrapperImp(ServletConfig config){
           this.config=config;
           this._servletname=this.config.getServletName();
           this.context=this.config.getServletContext();
    }
   
    public Servlet getServlet() throws ServletException{
 
                              destroy();
                              try {
                                            WebAppClassLoader loader=new WebAppClassLoader(this.getClass().getClassLoader()); //自定义class loader
                                            String name=getServletName(); //从ServletConfig中获得Servlet Name
                                            synchronized (context) {
                                                          Servlet theServlet=context.getServlet(name); //在ServletContext中查找Servlet
                                                          if(theServlet==null){         //如果ServletContext没有。              
                                                                        servletClass = loader.loadClass(name); //由Class loader 加载Servlet class。
                                                              theServlet = (Servlet) servletClass.newInstance(); //Servlet实例化。
                                                              WebAppContext.addServlet(name,theServlet); //将Servlet实例存入ServletContext。
                                                                        InvocationHandler handler=new ServletHandler(theServlet); //自定义ServletHandler,参见ServletHandler类。
                                                                        _theServlet=(Servlet)Proxy.newProxyInstance(theServlet.getClass().getClassLoader(),
                                                                                   new Class[]{Servlet.class},handler); //代理类实例化。
                                                                        _theServlet.init(config); //Servlet代理对象调用init方法。参见ServletHandler的invoke方法。
                                                                       
                                                          }else{ //ServletContext里已存在。
                                                                        InvocationHandler handler=new ServletHandler(theServlet); //自定义ServletHandler,参见ServletHandler类。
                                                                        _theServlet=(Servlet)Proxy.newProxyInstance(theServlet.getClass().getClassLoader(),
                                                                                   new Class[]{Servlet.class},handler);                 //代理Servlet接口,动态生成代理类,并实例化。                                  
                                                          }
                                            }
                                            return _theServlet; //返回Servlet代理对象
                              } catch( ClassNotFoundException ex1 ) {
                                
                              } catch( InstantiationException ex ) {
                                 
                              }catch(IllegalAccessException ex2){
                                           
                              }
                      return null;
    }
    public void destroy() {
        if (_theServlet != null) {
                      _theServlet.destroy();
        }
    }   
   
    protected String getServletName(){
           return _servletname;
    }
}
 
其中的ServletConfig保存Servlet相关信息。ServletContext保存所有的Servlet对象。WebAppClassLoader为自定义class loader,参见 http://blog.csdn.net/tyrone1979/archive/2006/09/03/1164262.aspx
 
最后编写测试类Main,该类模拟10个用户访问Servlet,5人访问DefaultServlet,5人访问UserServlet。
package org.colimas.main;
 
import java.io.IOException;
 
import javax.servlet.Servlet;
import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
 
import org.colimas.webapp.HttpServletRequestWrapper;
import org.colimas.webapp.HttpServletResponseWrapper;
import org.colimas.webapp.ServletConfigImpl;
import org.colimas.webapp.ServletWrapper;
import org.colimas.webapp.ServletWrapperImp;
import org.colimas.webapp.WebAppContext;
 
public class Main {
 
              private ThreadGroup _threadGroup;
              private Thread[] _threads;
              String defaultServletName="org.colimas.servlet.DefaultServlet";
              String userServletName="org.colimas.servlet.UserServlet";
              WebAppContext context=WebAppContext.newInstance();     
             
              public void doStart(){
             
                            _threadGroup=new ThreadGroup("SERVLETS");
                            int i=0;
                            _threads=new ServletThread[10]; //模拟10位用户。
                            for(i=0;i<5;i++){
                                          _threads[i]=new ServletThread(_threadGroup,new Integer(i).toString(),
                                                                      defaultServletName);
                                          _threads[i].start();
                            }
                            for(i=5;i<10;i++){
                                          _threads[i]=new ServletThread(_threadGroup,new Integer(i).toString(),
                                                                      userServletName);
                                          _threads[i].start();
                            }                          
                           
              }
              /**
               * @param args
               */
              public static void main(String[] args) {
                            Main _main=new Main();
                            _main.doStart();
                           
              }
              //模拟用户线程
              private class ServletThread extends Thread{
                           
                            private String servletName;
                           
                            public ServletThread(ThreadGroup group,String threadname,String servletname){
                                          super(group,threadname);
                                          servletName=servletname;
                            }
                            //调用Servlet的service.
                            public void run() { //用户调用Servlet
                                          ServletConfig config=new ServletConfigImpl(servletName,context); //调用的Servlet信息。
                                          ServletWrapperImp wrapper=new ServletWrapperImp(config);
                                          try {
                                                        Servlet defaultServlet=wrapper.getServlet(); //获得Servlet对象,实际是Servlet的代理对象
                                                        defaultServlet.service(new HttpServletRequestWrapper(),
                                                                                    new HttpServletResponseWrapper()); 调用代理对象的service方法,参见ServletHandler的invoke方法。
                                          } catch (ServletException e) {
                                                                                                                e.printStackTrace();
                                          } catch(IOException e){
                                                       
                                          }
                            }
                           
              }
}
HttpServletRequestWrapper和HttpServletResponseWrapper实现HttpServletRequest,和HttpServletResponse。
测试结果如下
org.colimas.servlet.DefaultServlet:Init servlet starting...
org.colimas.servlet.DefaultServlet:Running init
org.colimas.servlet.DefaultServlet:Init servlet ending...
org.colimas.servlet.UserServlet:Init servlet starting...
org.colimas.servlet.UserServlet:Running init
org.colimas.servlet.UserServlet:Init servlet ending...
org.colimas.servlet.DefaultServlet:service starting...
org.colimas.servlet.DefaultServlet:service ending...
org.colimas.servlet.DefaultServlet:service starting...
org.colimas.servlet.DefaultServlet:service ending...
org.colimas.servlet.UserServlet:service starting...
org.colimas.servlet.UserServlet:Do UserSErvlet Get
org.colimas.servlet.UserServlet:service ending...
org.colimas.servlet.UserServlet:service starting...
org.colimas.servlet.UserServlet:Do UserSErvlet Get
org.colimas.servlet.UserServlet:service ending...
org.colimas.servlet.UserServlet:service starting...
org.colimas.servlet.UserServlet:Do UserSErvlet Get
org.colimas.servlet.UserServlet:service ending...
org.colimas.servlet.UserServlet:service starting...
org.colimas.servlet.DefaultServlet:service starting...
org.colimas.servlet.UserServlet:service starting...
org.colimas.servlet.DefaultServlet:service starting...
org.colimas.servlet.DefaultServlet:service starting...
org.colimas.servlet.UserServlet:Do UserSErvlet Get
org.colimas.servlet.DefaultServlet:service ending...
org.colimas.servlet.UserServlet:Do UserSErvlet Get
org.colimas.servlet.DefaultServlet:service ending...
org.colimas.servlet.DefaultServlet:service ending...
org.colimas.servlet.UserServlet:service ending...
org.colimas.servlet.UserServlet:service ending...
 
2个Servlet第一次Load时初始化,被调用init,之后保存到ServletContext中。第二次直接从ServletContext获得,执行service。红字表示代理类里增加的输出结果。
 
4.动态代理的限制
              JDK的动态代理并不能随心所欲的代理所有的类。Proxy.newProxyInstance方法的第二个参数只能是接口数组, 也就是Proxy只能代理接口。

===================================================================================================================
首先写个普通的代理 

来个接口: 
Java代码   收藏代码
  1. public interface Drawable {  
  2.       
  3.     public void draw();  
  4.   
  5. }  

再来个实现类: 
Java代码   收藏代码
  1. public class Pen implements Drawable {  
  2.   
  3.     public void draw() {  
  4.         System.out.println("draw something");  
  5.     }  
  6.   
  7. }  


代理类(随便在博客写,未经编译): 
Java代码   收藏代码
  1. public class ProxyPen implements Drawable{  
  2. private Drawable draw;  
  3. public ProxyPen(Drawable draw){  
  4. this.draw = draw;  
  5. }  
  6. public viod draw(){  
  7. System.out.println("before draw");  
  8. draw.draw();  
  9. }  
  10. }  
  11. //接下来为调用代码  
  12. Drawable pen = new Pen();  
  13. ProxyPen proxy = new ProxyPen(pen);  
  14. proxy.draw();  

以上的静态代理很简单,无非就是聚合+多态。 
但是只能代理实现Drawable接口的类,显然具有很大局限性。 
而动态代理顾名思义就是动态生成代理类,ProxyPen将根据参数动态生成,要让其动态生成显然必须取得代理的接口,取得接口后能实现代理类,但是代理类添加的内容却还是定死的System.out.println("before draw");这也不满足动态要求,所以代理内容必须也要能自定义,想输任何代理内容,当然还是多态,传入一个处理代理内容的接口Handler即可: 
首先定义代理接口: 
Java代码   收藏代码
  1. public interface InvocationHandler {  
  2.         public void invoke(Method m,Object... args) throws Exception;  
  3. }  

Java代码   收藏代码
  1. public class Proxy {  
  2.   
  3.     public static Object newProxy(Class interfaces,InvocationHandler handler) throws Exception{  
  4.         String r = "\n";  
  5.           
  6.         Method[] methods = interfaces.getMethods();  
  7.           
  8.         StringBuffer sb = new StringBuffer("");  
  9.           
  10.         for(int i =0;i<methods.length;i++){  
  11.             sb.append(" public void "+methods[i].getName()+"() {"+r+  
  12.             "       try{ "+r+  
  13.             "       Method md = "+interfaces.getName()+".class.getMethod(\""+methods[i].getName()+"\");"+r+  
  14.             "       handler.invoke( md,new Object[]{});"+r+  
  15.             "       }catch(Exception e){e.printStackTrace();}"+r+     
  16.             "   }"+r  
  17.             );  
  18.         }  
  19.           
  20.         String src = "package com.javaeye.aronlulu.proxy;"+r+  
  21.                      "import java.lang.reflect.*;"+r+  
  22.         "public class ProxyPen implements "+interfaces.getName() +"{"+r+  
  23.       
  24.         "   private com.javaeye.aronlulu.proxy.InvocationHandler handler;"+r+  
  25.       
  26.         "   public ProxyPen("+handler.getClass().getName()+" handler){"+r+  
  27.         "       this.handler = handler;"+r+  
  28.         "   }"+r+  
  29. System.out.println(\"before\");"+r+  
  30.   
  31.         sb.toString()+  
  32.   
  33.         "}" +r;  
  34.         String dir = System.getProperty("user.dir")+"/src/com/javaeye/aronlulu/proxy/";  
  35.         FileWriter writer = new FileWriter(new File(dir+"ProxyPen.java"));  
  36.         writer.write(src);  
  37.         writer.flush();  
  38.         writer.close();  
  39.         //编译动态代理类  
  40.         JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();  
  41.         StandardJavaFileManager fileMgr = compiler.getStandardFileManager(nullnullnull);  
  42.         Iterable units = fileMgr.getJavaFileObjects(dir+"ProxyPen.java");  
  43.         CompilationTask t = compiler.getTask(null, fileMgr, nullnullnull, units);  
  44.         t.call();  
  45.         fileMgr.close();  
  46.                   //加载并调用返回代理接口  
  47.         URL[] urls = new URL[]{new URL("file:/"+dir)};  
  48.         URLClassLoader loader = new URLClassLoader(urls);  
  49.         Class c = loader.loadClass("com.javaeye.aronlulu.proxy.ProxyPen");  
  50.           
  51.         Constructor ctr = c.getConstructor(handler.getClass());  
  52.         return ctr.newInstance(handler);  
  53.           
  54.     }  
  55.       
  56. }  


接下来编写具体处理代理类代码: 
Java代码   收藏代码
  1. public class DrawHandler implements InvocationHandler {  
  2.       
  3.     private Object draw;  
  4.       
  5.     public DrawHandler (Object draw){  
  6.         this.draw = draw;  
  7.     }  
  8.   
  9.     public void invoke(Method m,Object[] args) throws Exception{  
  10.         System.out.println("before");  
  11.         m.invoke(draw, args);  
  12.         System.out.println("after");  
  13.     }  
  14.   
  15. }  

测试代码: 
Java代码   收藏代码
  1. Drawable drawable = new Pen();  
  2. InvocationHandler handler = new DrawHandler(drawable);  
  3. Drawable draw = (Drawable) Proxy.newProxy(Drawable.class,handler);  
  4. draw.draw();  

动态代理的实现简单完成了,当然跟java的动态代理实现没法比,中间还封装了很多类型检查及反射实现。动态代理原理其实无非就是反射+多态+聚合的实现,当然有了动态代理,很多强大的功能就能实现了,最有名的无非就是AOP,java动态代理的局限性是只能针对接口,若要针对具体业务类,则需要CGLIB这样的第三方框架了,springAOP则采用了以上两种组合,默认还是使用java动态代理,设置Bean中一个属性开关来控制,若非接口实现,则开始调用CGLIB。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值