手動寫源碼---模擬實現動態代理

AOP的實現有兩種,一種是接口的實現,一種是產生本身實現,分別的表明爲JDK的Proxy和CGLIB的Proxyhtml

下面是模擬接口代理實現 ,經過模擬JDK的動態代理,更深入的理解java

經過動態代理能夠面向切面編程spring

1 定義被代理的對象和接口apache

Java代碼   收藏代碼
  1. public interface BookInterface {  
  2.     void selling();  
  3. }  

 

Java代碼   收藏代碼
  1. public interface PhoneInterface {  
  2.     void selling();  
  3. }  

 

Java代碼   收藏代碼
  1. public class Book implements BookInterface {  
  2.     @Override  
  3.     public void selling() {  
  4.         System.out.println("books selling.....");  
  5.     }  
  6. }  

 

Java代碼   收藏代碼
  1. public class VivoPhone implements PhoneInterface {  
  2.     @Override  
  3.     public void selling() {  
  4.         System.out.println("selling vivo x5");  
  5.     }  
  6. }  

 

 2 定義切面類編程

Java代碼   收藏代碼
  1. public class TimeAspect {  
  2.     static long bgn;  
  3.     public static void before(){  
  4.         bgn = System.currentTimeMillis();  
  5.         System.out.println("begin time...  " + bgn);  
  6.     }  
  7.     public static void after(){  
  8.         long end = System.currentTimeMillis();  
  9.         System.out.println("end time...  " + (end-bgn));  
  10.     }  
  11. }  
Java代碼   收藏代碼
  1. public class LogAspect{  
  2.     public static void before(){  
  3.            
  4.         System.out.println("begin log...");  
  5.     }  
  6.     public static void after(){  
  7.         System.out.println("finish log...");  
  8.     }  
  9. }  

 3 定義InvocationHander  代理接口ide

Java代碼   收藏代碼
  1. import java.lang.reflect.Method;  
  2. public interface InvocationHander {  
  3.     public void invoke(Object o,Method m);  
  4. }  

   代理類(切面編程裏面也能夠作一些特殊的處理)函數

Java代碼   收藏代碼
  1. import java.lang.reflect.Method;  
  2. import jdkproxy.LogTranService;  
  3. import jdkproxy.TimeTranService;  
  4.   
  5. public class ProxyHander implements InvocationHander {  
  6.     private Object target;  
  7.     public ProxyHander(Object target) {  
  8.         this.target = target;  
  9.     }  
  10.     @Override  
  11.     public void invoke(Object o, Method m) {  
  12.         try {  
  13.             TimeTranService.before();  
  14.             if(!(o instanceof BookInterface)){//只有非BookInterface接口調用日誌  
  15.                 LogTranService.before();  
  16.             }  
  17.               
  18.             m.invoke(target);  
  19.                
  20.             if(!(o instanceof BookInterface)){  
  21.                 LogTranService.after();  
  22.             }  
  23.             TimeTranService.after();  
  24.         } catch (Exception e) {  
  25.             e.printStackTrace();  
  26.         }   
  27.     }  
  28. }  

    動態代理類測試

Java代碼   收藏代碼
  1. import java.io.File;  
  2. import java.io.IOException;  
  3. import java.lang.reflect.Constructor;  
  4. import java.lang.reflect.Method;  
  5.   
  6. import javax.tools.JavaCompiler;  
  7. import javax.tools.JavaCompiler.CompilationTask;  
  8. import javax.tools.StandardJavaFileManager;  
  9. import javax.tools.ToolProvider;  
  10.   
  11.   
  12. import org.apache.commons.io.FileUtils;  
  13.   
  14. public class Proxy {  
  15.     /* 
  16.      * 空的構造函數 
  17.      */  
  18.     private Proxy(){  
  19.     }  
  20.     /* 
  21.      * 返回代理類 
  22.      */  
  23.     public static Object newProxyInstance(Class inter,InvocationHander h){  
  24.         String packageName  = inter.getPackage().getName();  
  25.         // String temp = new String(packageName);  
  26.         String proxyClassName = "$"+packageName.replace(".","_") + "_" + inter.getSimpleName() +"Proxy";  
  27.           
  28.         String InHanderPackage = h.getClass().getPackage().getName();  
  29.         String rt = "\r\n";// 換行  
  30.         String methodCode = "";  
  31.         for (Method method:inter.getMethods()) {  
  32.             methodCode+="   @Override"+rt+  
  33.               
  34.             "   public void "+ method.getName()+"() {"+rt+  
  35.             "       try{"+rt+  
  36.             "           Method method  = "+inter.getName()+".class.getMethod(\""   
  37.             +           method.getName()+   "\");"+rt+  
  38.             "           h.invoke(this,method);      "+rt+  
  39.             "       }catch(Exception e ){" +rt+  
  40.             "           e.printStackTrace();" +rt+  
  41.             "       }"+rt+  
  42.             "   }";  
  43.         }  
  44.         /* 
  45.          * 總的java代碼 
  46.          */  
  47.         String javaCode=    
  48.         "package  "+packageName+";"+rt+  
  49.         "import "+InHanderPackage+".InvocationHander;"+rt+  
  50.           
  51.         "import java.lang.reflect.Method;"+rt+  
  52.         "public class "+proxyClassName+" implements "+inter.getName()+" {"+rt+  
  53.         "   public "+proxyClassName+"("+InHanderPackage+".InvocationHander h) {"+rt+  
  54.         "       super();"+rt+  
  55.         "       this.h = h;"+rt+  
  56.         "   }"+rt+  
  57.         "   private "+InHanderPackage+".InvocationHander h;"+rt+  
  58.         methodCode+rt+  
  59.         "}";  
  60.         /* 
  61.          *  生成java文件 
  62.          */  
  63.         // 生成文件路徑  
  64.         String filename =  System.getProperty("user.dir")+"/bin/"+packageName.replace(".", "//")+"/"+proxyClassName+".java";  
  65.         File file = new File(filename);  
  66.         try {  
  67.             // 須要commons-io的jar方便的操做文件  
  68.             FileUtils.writeStringToFile(file, javaCode);   
  69.         } catch (IOException e) {  
  70.             e.printStackTrace();  
  71.         }  
  72.           
  73.           
  74.         // 編譯  拿到編譯器  
  75.         JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();  
  76.         // 文件管理  
  77.         StandardJavaFileManager fileMgr = compiler.getStandardFileManager(null, null, null);  
  78.         //獲取文件  
  79.         Iterable units = fileMgr.getJavaFileObjects(filename);  
  80.           
  81.         // 編譯任務  
  82.         CompilationTask t  = compiler.getTask(null, fileMgr, null, null, null, units);  
  83.         // call進行編譯  
  84.         t.call();  
  85.         try {  
  86.             fileMgr.close();  
  87.         } catch (IOException e) {  
  88.             e.printStackTrace();  
  89.         }  
  90.           
  91.         // load到內存  
  92.         ClassLoader cl = ClassLoader.getSystemClassLoader();  
  93.         try {  
  94.             Class c = cl.loadClass(packageName+"."+proxyClassName);  
  95.             Constructor ctr = c.getConstructor(InvocationHander.class);  
  96.             return ctr.newInstance(h);  
  97.         } catch (Exception e) {  
  98.             e.printStackTrace();  
  99.         }  
  100.         return null;  
  101.     }  
  102. }  

4 測試ui

Java代碼   收藏代碼
  1. import org.junit.After;  
  2. import org.junit.Before;  
  3. import org.junit.Test;  
  4. public class TestProxy {  
  5.     @Before  
  6.     public void  before(){  
  7.         System.out.println("-----------------start-------------------");  
  8.     }  
  9.     @Test  
  10.     public  void test() {  
  11.         Book book = new Book();  
  12.         InvocationHander h = new ProxyHander(book);  
  13.         BookInterface bi = (BookInterface)Proxy.newProxyInstance(BookInterface.class,h);  
  14.         bi.selling(); // 沒有日誌打印  
  15.           
  16.         System.out.println("==================分割==============================");  
  17.           
  18.         PhoneInterface car = new VivoPhone();  
  19.         h = new ProxyHander(car);  
  20.         PhoneInterface pi = (PhoneInterface)Proxy.newProxyInstance(PhoneInterface.class,h);  
  21.         pi.selling(); // 有日誌打印  
  22.     }  
  23.     @After  
  24.     public void  after(){  
  25.         System.out.println("-----------------end-------------------");    
  26.     }  
  27. }  

 Proxy類裏面生成代理類名稱的方法是根據包名來的,全類名長度加起來超過250多個長度可能會讓java類沒法編譯,那就須要特殊處理了。超過250個長度的全類名那種項目沒見過,不考慮this

上面是他的原理,spring中如何使用aop呢?

 請參考

 參考文章  

相關文章
相關標籤/搜索