動態代理

案例

獲取1個方法的運行時間

  • 使用繼承實現(繼承)java

  • 使用實現接口實現(組合)設計模式

//繼承
public class Tank {
    public void move() {
        System.out.println("Tank is moving ~~~");
    }
}
//繼承基類重寫裏面的方法對其進行擴展(對Tank類的move()方法加上獲取運行時間的代碼)
public class TimeHandlerTank extends Tank {
    @Override
    public void move() {
        long start = System.currentTimeMillis();
        super.move();
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        long end = System.currentTimeMillis();
        System.out.println("move時間: " + String.valueOf(end - start));
    }
}

public static void main(String[] args) {
    new TimeHandlerTank().move();
}
//結果
Tank is moveing~~~
move時間: 1003

//組合
public interface Moveable {
    void move();
}

public class Tank implements Moveable {

    public void move() {
        System.out.println("Tank is moveing~~~");
    }
}
//經過構造方法傳入Tank
public class TimeHandler implements Moveable{
    private Moveable moveable ;

    public TimeHandler(Moveable moveable) {
        this.moveable = moveable;
    }

//實現move()方法的時候,調用Tank的move()方法,同時加上獲取運行時間的代碼
    public void move() {
        long start = System.currentTimeMillis();
        moveable.move();
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        long end = System.currentTimeMillis();
        System.out.println("move時間: " + String.valueOf(end - start));
    }
}

public static void main(String[] args) {
    Moveable tank = new Tank();
    new TimeHandler(tank).move();
}

獲取一個方法的執行時間同時記錄日誌

使用繼承的方式就是繼續繼承重寫方法,這樣的話就會出現好多類(類爆炸),若是使用實現接口的方式(組合)將會避免這個問題。
//組合實現,對於上面組合的代碼添加LogHandler類
public class LogHandler implements Moveable{
    private Moveable moveable;

    public LogHandler(Moveable moveable) {
        this.moveable = moveable;
    }

    public void move() {
        moveable.move();
        System.out.println("記錄日誌~~~~");
    }
}
//main方法修改以下:
public static void main(String[] args) {
    Moveable tank = new Tank();
    new LogHandler(new TimeHandler(tank)).move();
}
結果:
Tank is moveing~~~
move時間: 1005
記錄日誌~~~~

jdk動態代理

雖然能夠經過編寫hanler來擴展類,可是這樣的話會產生的大量的類,如何能達到這樣的效果又不產生大量的類呢 ,其實經過反射+動態編譯即可以解決。其實jdk就是這樣實現的,下面是jdk動態代理的例子:
dom

//定義接口
public interface ForumService {
    void removeTopic(int topicId);
    void removeForum(int forumId);
}
//實現接口
public class ForumServiceImpl implements ForumService {
    public void removeTopic(int topicId) {
        System.out.println("模擬刪除topic記錄:" + topicId);
        try {
            Thread.sleep(100);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    public void removeForum(int forumId) {
        System.out.println("模擬刪除formId:" + forumId);
        try {
            Thread.sleep(200);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}
//運行時間代碼(橫切代碼)
public class PerformanceMonitor {

    public static void begin(String method){
        System.out.println("begin monitor...");
    }

    public static void end(){
        System.out.println("end monitor");
    }
}
//將目標業務類和橫切代碼編制在一塊兒
public class PerformanceHandler implements InvocationHandler {
    private Object target;

    public PerformanceHandler(Object target) {
        this.target = target;
    }

    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        PerformanceMonitor.begin(target.getClass().getName() + "." + method.getName());
        Object obj = method.invoke(target, args);
        PerformanceMonitor.end();
        return obj;
    }
}
//
public class TestForumService {
    public static void main(String[] args) {
        //但願被代理的目標類
        ForumService target = new ForumServiceImpl();
        //將目標業務類和橫切代碼編制在一塊兒
        PerformanceHandler handler = new PerformanceHandler(target);
        //返回代理對象
        ForumService proxy = (ForumService) Proxy.newProxyInstance(target.getClass().getClassLoader(),
                target.getClass().getInterfaces(), handler);

        proxy.removeForum(10);
        System.out.println("*****************");
        proxy.removeTopic(120);
    }
}

//結果:
begin monitor...
模擬刪除formId:10
end monitor
*****************
begin monitor...
模擬刪除topic記錄:120
end monitor

模擬實現jdk動態代理

//定義被代理對象的類必須實現的接口
public interface Moveable {
   void move();
}
//被代理對象實現接口
public class Tank implements Moveable {

   public void move() {
      
      System.out.println("Tank Moving...");
      try {
         Thread.sleep(new Random().nextInt(10000));
      } catch (InterruptedException e) {
         e.printStackTrace();
      }
   }
}
//反射調用被代理對象接口
public interface InvocationHandler {
   public void invoke(Object o, Method m);
}
//實現反射調用被代理對象接口
public class TimeHandler implements InvocationHandler {

    private Object target;

    public TimeHandler(Object target) {
        super();
        this.target = target;
    }

    public void invoke(Object o, Method m) {
        long start = System.currentTimeMillis();
        System.out.println("starttime:" + start);
        System.out.println(o.getClass().getName());
        try {
            m.invoke(target);
        } catch (Exception e) {
            e.printStackTrace();
        }
        long end = System.currentTimeMillis();
        System.out.println("time:" + (end - start));
    }

}
//產生代理對象
public class Proxy {
   public static Object newProxyInstance(Class infce, InvocationHandler h) throws Exception { //JDK6 Complier API, CGLib, ASM
      String methodStr = "";
      String rt = "\r\n";
      
      Method[] methods = infce.getMethods();
      /*
      for(Method m : methods) {
         methodStr += "@Override" + rt + 
                   "public void " + m.getName() + "() {" + rt +
                     "   long start = System.currentTimeMillis();" + rt +
                     "   System.out.println(\"starttime:\" + start);" + rt +
                     "   t." + m.getName() + "();" + rt +
                     "   long end = System.currentTimeMillis();" + rt +
                     "   System.out.println(\"time:\" + (end-start));" + rt +
                   "}";
      }
      */
      for(Method m : methods) {
         methodStr += rt +
                   "public void " + m.getName() + "() {" + rt +
                   "    try {" + rt +
                   "    Method md = " + infce.getName() + ".class.getMethod(\"" + m.getName() + "\");" + rt +
                   "    h.invoke(this, md);" + rt +
                   "    }catch(Exception e) {e.printStackTrace();}" + rt +
                  
                   "}";
      }
      
      String src = 
         "package com.bjsxt.proxy;" +  rt +
         "import java.lang.reflect.Method;" + rt +
         "public class $Proxy1 implements " + infce.getName() + "{" + rt +
         "    public $Proxy1(InvocationHandler h) {" + rt +
         "        this.h = h;" + rt +
         "    }" + rt +
         
         
         "    com.bjsxt.proxy.InvocationHandler h;" + rt +
                     
         methodStr +
         "}";
      String fileName = 
         "/Users/wjk/myproject/test/design_pattern/src/main/java/com/bjsxt/proxy/$Proxy1.java";
      File f = new File(fileName);
      FileWriter fw = new FileWriter(f);
      fw.write(src);
      fw.flush();
      fw.close();
      
      //compile(Java動態編譯)
      JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
      StandardJavaFileManager fileMgr = compiler.getStandardFileManager(null, null, null);
      Iterable units = fileMgr.getJavaFileObjects(fileName);
      CompilationTask t = compiler.getTask(null, fileMgr, null, null, null, units);
      t.call();
      fileMgr.close();
      
      //load into memory and create an instance
      URL[] urls = new URL[] {new URL("file:/" + "/Users/wjk/myproject/test/design_pattern/src/main/java")};
      URLClassLoader ul = new URLClassLoader(urls);
      Class c = ul.loadClass("com.bjsxt.proxy.$Proxy1");
      System.out.println(c);
      
      Constructor ctr = c.getConstructor(InvocationHandler.class);
      Object m = ctr.newInstance(h);
      //m.move();

      return m;
   }
}

public class Client {
   public static void main(String[] args) throws Exception {
      Tank t = new Tank();
     
      InvocationHandler h = new TimeHandler(t);
      
      Moveable m = (Moveable)Proxy.newProxyInstance(Moveable.class, h);
      
      m.move();
   }
}
//結果
class com.bjsxt.proxy.$Proxy1
starttime:1452505387375
com.bjsxt.proxy.$Proxy1
Tank Moving...
time:9798

參考資料:ide

《Spring3.x企業應用開發實踐》this

《馬士兵——設計模式》url

相關文章
相關標籤/搜索