国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 學院 > 開發設計 > 正文

springAOP的兩種代理

2019-11-14 15:20:27
字體:
來源:轉載
供稿:網友

本篇記錄下sPRing AOP的兩種代理,為下一篇AOP實現做下鋪墊。

1.JDK動態代理  2.cglib代理

1、如果目標對象實現了接口,默認情況下會采用JDK的動態代理實現AOP
2、如果目標對象實現了接口,可以強制使用CGLIB實現AOP
3、如果目標對象沒有實現了接口,必須采用CGLIB庫,spring會自動在JDK動態代理和CGLIB之間轉換

:JDK動態代理要比cglib代理執行速度快,但性能不如cglib好。所以在選擇用哪種代理還是要看具體情況,一般單例模式用cglib比較好,具體原因請自行百度。

一。JDK動態代理實現(原理是使用反射機制)

首先定義接口,并實現

      public interface TestService{

           public int add();   
      }

      public class TestServiceImpl implements TestService{

         @Override
         public int add() {
            System.out.println("開始執行add...");
            return 0;
         }
     }

 

定義代理類,這里要注意導入的包是import java.lang.reflect.*

   public class JDKDynamicProxy implements InvocationHandler {

     //被代理的目標對象
    private Object proxyObj;  
   
    /**
      * Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)
      * loader    :類加載器 一個ClassLoader對象,定義了由哪個ClassLoader對象來對生成的代理對象進行加載
      * interfaces:一個Interface對象的數組,表示的是我將要給我需要代理的對象提供一組什么接口,如果我提供了一組接口給它,那么這個代理對象就宣稱實現了該接口(多態),這樣我就能調用這組接口中的方法了
       * h         :一個InvocationHandler對象,表示的是當我這個動態代理對象在調用方法的時候,會關聯到哪一個InvocationHandler對象上
     */
       public Object newProxy(Object proxyObj){  
             this.proxyObj = proxyObj;
            //返回一個代理對象  
           return Proxy.newProxyInstance(proxyObj.getClass().getClassLoader(),   
                                      proxyObj.getClass().getInterfaces(),   
                                      this);  
       }  

      /**
       * 執行目標對象
       * Object  proxy:被代理的對象
       * Method  method:要調用的方法
       * Object  args[]:方法調用時所需要的參數
       */
        @Override
        public Object invoke(Object proxy, Method method, Object[] args)
                                   throws Throwable {      
             before();
             Object object = method.invoke(this.proxyObj,args);  // 通過反射機制調用目標對象的方法
             after();      
             return object;  
         }
    
         public void before(){
              System.out.println("開始執行目標對象之前...");
         }
    
         public void after(){
             System.out.println("開始執行目標對象之后...");
         }
     }

測試類:

      public static void main(String[] args) {
        
          //我們要代理的真實對象
          TestService testService = new TestServiceImpl();       
          //testJDKProxyService.add();//不是用代理   
        
         JDKDynamicProxy JDKDynamicProxyTarget = new JDKDynamicProxy();
         TestService testServiceProxy = (TestService) JDKDynamicProxyTarget.newProxy(testService);
         //執行代理類的方法  
         testServiceProxy.add();

     }

控制臺顯示

   

 

二。CGLIB代理,需要導入 cglib-nodep-2.1_3.jar

    先說下cglib,CGlib是一個強大的,高性能,高質量的Code生成類庫。它可以在運行期擴展Java類與實現Java接口。

先定義一個實現類(注意并沒有實現接口)

  public class TestCGLIBServiceImpl {

    public int add() {
        System.out.println("開始執行add...");
        return 0;
    }
 }

定義cglib代理類,此時導入的包應該是import net.sf.cglib.proxy.*

  import java.lang.reflect.Method;
  import net.sf.cglib.proxy.Enhancer;
  import net.sf.cglib.proxy.MethodInterceptor;
  import net.sf.cglib.proxy.MethodProxy;

    public class CGLIBProxy implements MethodInterceptor{

    private Object targetObject ;//被代理的目標對象
    
    public Object createProxyInstance(Object targetObject) {

           this . targetObject = targetObject;

           Enhancer enhancer = new Enhancer();

           enhancer.setSuperclass(targetObject.getClass());// 設置代理目標

           enhancer.setCallback( this );// 設置回調

           return enhancer.create();

    }
    

    /**
     * 在代理實例上處理方法調用并返回結果
     * @param object : 代理類
     * @param method :被代理的方法
     * @param args :該方法的參數數組
     * @param methodProxy
     */
    @Override
    public Object intercept(Object object, Method method, Object[] args,
            MethodProxy methodproxy) throws Throwable {       
        Object result = null;    
        try {
              System. out .println( "前置處理開始 ..." );
              result = methodproxy.invoke( targetObject , args);//執行目標對象的方法
              System. out .println( "后置處理開始  ..." );
           } catch (Exception e) {
               System. out .println( " 異常處理 ..." );
           } finally {
               System. out .println( " 調用結束 ..." );
           }
           return result;
       }  
   }
測試類:

   public class TestCGLIBProxy {

        public static void main(String[] args) {
        
          //我們要代理的真實對象
          TestCGLIBServiceImpl testCGLIB = new TestCGLIBServiceImpl();
           CGLIBProxy CGLIBproxy = new CGLIBProxy();
           TestCGLIBServiceImpl testCGLIBProxy = (TestCGLIBServiceImpl) CGLIBproxy.createProxyInstance(testCGLIB);
           testCGLIBProxy.add();
       }
   }

結果圖:

  

寫在后面:spring AOP的兩種代理實現代碼就寫到這,這里只是實現了,如果你要想真正明白,還得熟悉其中原理機制,比如反射機制,newProxyInstance(...),Enhancer()原理,invoke()原理等等。

下一篇將要記錄spring AOP的基于注解的具體實現,而對于xml的我姑且不是用了,因為不夠靈活,不夠強大,要是有時間還會介紹下spring AOP 自定義注解的實現

 


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 龙江县| 甘南县| 上林县| 普格县| 班戈县| 来宾市| 保靖县| 长阳| 金山区| 柳林县| 武威市| 梅河口市| 突泉县| 河南省| 荆州市| 平舆县| 尼勒克县| 黄梅县| 疏勒县| 永丰县| 曲靖市| 中山市| 曲周县| 和平区| 九台市| 大安市| 佛教| 梁平县| 湖口县| 福建省| 泰州市| 廊坊市| 成安县| 太康县| 阿巴嘎旗| 正安县| 平谷区| 玉林市| 大冶市| 绥化市| 唐山市|