使用繼承實現(繼承)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 記錄日誌~~~~
雖然能夠經過編寫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
//定義被代理對象的類必須實現的接口 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