亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java使用JDK與Cglib動態(tài)代理技術(shù)統(tǒng)一管理日志記錄

 更新時間:2020年05月06日 10:25:30   作者:藍山.  
這篇文章主要介紹了Java使用JDK與Cglib動態(tài)代理技術(shù)統(tǒng)一管理日志記錄,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

Java中動態(tài)代理主要有JDK和CGLIB兩種方式。

區(qū)別主要是jdk是代理接口,而cglib是代理類。

  • 優(yōu)點:這種方式已經(jīng)解決我們前面所有日記需要的問題。非常的靈活。而且可以方便的在后期進行維護和升級。
  • 缺點:當然使用jdk動態(tài)代理,必需要有接口。如果沒有接口。就無法使用jdk動態(tài)代理技術(shù)。

計算接口 Calculate.java

public interface Calculate {
 /**
  * 加法運算
  * @param num1 參數(shù) 1
  * @param num2 參數(shù) 2
  * @return
  */
 public int add(int num1, int num2);

 /**
  * 加法運算
  * @param num1 參數(shù) 1
  * @param num2 參數(shù) 2
  * @param num3 參數(shù) 3
  * @return
  */
 public int add(int num1, int num2, int num3);

 /**
  * 除法運算
  * @param num1 參數(shù) 1
  * @param num2 參數(shù) 2
  * @return
  */
 public int div(int num1, int num2);
}

實現(xiàn)計算接口中的方法 CalculateImpl.java

/**
 * 實現(xiàn)計算接口中的方法
 * Created by YongXin Xue on 2020/05/05 11:29
 */
public class CalculateImpl implements Calculate {
 @Override
 public int add(int num1, int num2) {
  // 記錄當前操作,及運算參數(shù)
  LogUtils.logBefore("add", num1, num2);
  int result = num1 + num2;
  return result;
 }

 @Override
 public int add(int num1, int num2, int num3) {
  // 記錄當前操作,及運算參數(shù)
  LogUtils.logBefore("add", num1, num2, num3);
  int result = num1 + num2 + num3;
  return result;
 }

 @Override
 public int div(int num1, int num2) {
  // 記錄當前操作,及運算參數(shù)
  LogUtils.logBefore("div", num1, num2);
  int result = 0;
  try {
   result = num1 / num2;
   // 記錄運算結(jié)果
   LogUtils.logAfterReturning("div", result);
  }catch (Exception e){
   // 記錄異常信息
   LogUtils.logAfterThrowing("div", e);
  }
  return result;
 }
}

記錄日志工具類 LogUtils.java

/**
 * 記錄日志工具類
 * Created by YongXin Xue on 2020/05/05 11:38
 */
public class LogUtils {
 /**
  * 記錄前置的日志操作
  * @param method 當前運算操作
  * @param args 當前運算參數(shù)
  */
 public static void logBefore(String method, Object ... args){
  System.out.println("操作運算是 : " + method + " 參數(shù)是 : " + Arrays.asList(args));
 }

 /**
  * 返回日志操作
  * @param method 當前方法
  * @param result 當前操作返回值
  */
 public static void logAfterReturning(String method, Object result){
  System.out.println("當前操作運算時 : " + method + " 返回值是 : " + result);
 }

 /**
  * 當前操作產(chǎn)生的異常
  * @param method 當前操作
  * @param e 發(fā)生的異常
  */
 public static void logAfterThrowing(String method, Exception e){
  System.out.println("當前運算時 : " + method + " 發(fā)生的異常是 : " + e);
 }
}

JDK 動態(tài)代理的工廠類 JDKProxyFactory.java

/**
 * JDK 動態(tài)代理的工廠
 * Created by YongXin Xue on 2020/05/05 13:02
 */
public class JDKProxyFactory {

 /**
  * 通過 JDK 底層自帶的 JDK 動態(tài)代理技術(shù)解決日志需求問題
  * @param target
  * @return
  */
 public static Object createJDKProxy(Object target){
  /**
   * Proxy 是Jdk中自帶的一個工具類(反射包下,屬于反射的功能).
   * Proxy類的作用: 它可以幫我們創(chuàng)建代理類或?qū)嵗?
   * 方法newProxyInstance()說明: 創(chuàng)建代理對象實例
   * 第一個參數(shù)是: 目標對象的類加載器
   * 第二個參數(shù)是: 目標對象實現(xiàn)的所有接口
   * 第三個參數(shù)是: InvocationHandler 接口的實例
   * InvocationHandler 接口的實現(xiàn)類可以對代理的目標對象方法進行增強操作.
   * 代理的目標對象 ===>>> 需要額外增加功能的類(對象實例)
   * 增強操作 ===>>> 給原來功能添加的額外功能叫增強操作 ( 日記就是增強操作 )
   */
  return Proxy.newProxyInstance(
    target.getClass().getClassLoader(),
    target.getClass().getInterfaces(),
    new InvocationHandler() { // 匿名內(nèi)部類
     /**
      * invoke 方法是 InvocationHandler 接口中唯一的方法
      * 代理對象每次調(diào)用方法時,都會執(zhí)行 invoke() 方法 , 所有的增強操作都需要在invoke()方法中完成
      * @param proxy  代理對象實例
      * @param method 代理調(diào)用的方法的反射 Method 對象實例
      * @param args  調(diào)用代理方法時傳遞進來的參數(shù)
      * @return
      * @throws Throwable
      */
     @Override
     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
      System.out.println("代理調(diào)用了 invoke 方法 ");
      System.out.println(method);  //打印方法信息
      System.out.println(Arrays.asList(args)); //打印參數(shù)信息
      // invoke() 方法執(zhí)行代理對象的(加法 / 除法 / 增強日志)操作
      Object result = null;
      LogUtils.logBefore(method.getName(), args);
      try {
       // 1. 返回值是 method 方法調(diào)用時的返回值
       result = method.invoke(target, args);
       // 2. 增強操作
       LogUtils.logAfterReturning(method.getName(), result);
      }catch (Exception e){
       LogUtils.logAfterThrowing(method.getName(), e);
      }
      // invoke() 返回代理方法的返回值
      return result;
     }
    });
 }

 // 測試代碼
 public static void main(String[] args) {
  // 目標對象
  Calculate target = new CalculateImpl();
  // 創(chuàng)建 Calculate 的代理對象實例
  Calculate calculateProxy = (Calculate) createJDKProxy(target );
  // jdk動態(tài)代理對象實例和目標對象實例 同宗同族 ( 他們都實現(xiàn)了相同的接口 )
  System.out.println(calculateProxy instanceof Calculate);
  System.out.println(target instanceof Calculate);

  System.out.println( "代理方法的結(jié)果是 : " + calculateProxy.div(100,20) );

  // jdk動態(tài)代理創(chuàng)建出來的代理對象實例 是 目標對象 接口的一個實現(xiàn)類
  // 這個代理對象 和 目標對象類沒有父子關(guān)系 ( 只能用接口接收代理對象 )
 }
}

使用 Cglib 代理

  1. Jdk動態(tài)代理是通過實現(xiàn)目標對象所有接口產(chǎn)生一個代理對象實例從而解決問題.
  2. 如果目標對象沒有接口.則可以使用Cglib動態(tài)代理技術(shù).
  3. Cglib動態(tài)代理技術(shù)對目標對象有沒有實現(xiàn)接口,沒有要求.
  4. Cglib動態(tài)代理技術(shù),是通過拷貝然后修改目標對象的類的字節(jié)碼來產(chǎn)生一個代理對象
  5. 而且這個Cglib產(chǎn)生的代理對象實例 是 目標對象的一個子類.

IA 接口 IA.java

public interface IA {
 public String show(String start);
}

IA 實現(xiàn)類 IAImpl.java

public class IAImpl implements IA {
 @Override
 public String show(String start) {
  System.out.println(start + "開始表演!");
  return start + "表演的不錯!!";
 }
}

使用 Cglib 代理 CglibProxyFactory.java

/**
 * 使用 Cglib 代理
 * Created by YongXin Xue on 2020/05/05 15:03
 */
public class CglibProxyFactory {

 public static Object createCglibProxy(Object target){
  // 是 Cglib 用于創(chuàng)建代理對象的增強工具類
  Enhancer enhancer = new Enhancer();
  // Cglib需要對目標對象的Class字節(jié)碼進行修改.
  // Cglib產(chǎn)生的代理對象實例.是目標對象的子類
  enhancer.setSuperclass(target.getClass());
  // 只要是代理都會對原來的內(nèi)容進行增強操作 ( 增強就是在原有功能上 額外添加的功能 )
  // setCallback() 設(shè)置用于增強 操作的實現(xiàn)類( MethodInterceptor對代理方法進行攔截 )
  // 每次只要調(diào)用Cglib代理的方法,都會執(zhí)行 MethodInterceptor 接口中 intercept() 方法
  enhancer.setCallback(new MethodInterceptor() {
   /**
    * intercept() 方法 跟 JDK 代理中的 InvocationHandler接口中 invoke() 功能完全一樣
    * @param proxy  Cglib代理對象實例
    * @param method 調(diào)用方法的反射對象實例
    * @param args 調(diào)用方法時傳遞的參數(shù)
    * @param methodProxy 代理方法的method代理對象
    * @return  是代理對象方法的返回值.
    * @throws Throwable
    */
   @Override
   public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
    Object result = null;
    try {
     LogUtils.logBefore(method.getName(), args);
     // 調(diào)用目標方法 [加 / 減 / 乘 / 除 / 或具體方法]
     result = method.invoke(target, args);
     // 執(zhí)行增強代碼
     LogUtils.logAfterReturning(method.getName(), args);
    }catch (Exception e){
     e.printStackTrace();
     LogUtils.logAfterThrowing(method.getName(), e);
    }

    return result;
   }
  });
  // 創(chuàng)建 Cglib 代理對象實例
  return enhancer.create();
 }
 //測試
 public static void main(String[] args) {
  // 目標對象
  Calculate calculate = new CalculateImpl();
  // 創(chuàng)建代理對象實例
  Calculate cglibProxy = (Calculate) createCglibProxy(calculate);
  // 調(diào)用代理方法式會執(zhí)行 MethodInterceptor 接口中 intercept() 方法
  int result = cglibProxy.div(120, 0);
  // Cglib 代理 是目標子類執(zhí)行 MethodInterceptor 接口中 intercept() 方法
  System.out.println(cglibProxy instanceof Calculate);
 }
}

優(yōu)點:在沒有接口的情況下,同樣可以實現(xiàn)代理的效果。
缺點:同樣需要自己編碼實現(xiàn)代理全部過程。

到此這篇關(guān)于Java使用JDK與Cglib動態(tài)代理技術(shù)統(tǒng)一管理日志記錄的文章就介紹到這了,更多相關(guān)Java動態(tài)代理統(tǒng)一管理日志 內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 基于zipoutputStream的簡單使用

    基于zipoutputStream的簡單使用

    這篇文章主要介紹了基于zipoutputStream的簡單使用方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-12-12
  • Java sleep方法及中斷方式、yield方法代碼實例

    Java sleep方法及中斷方式、yield方法代碼實例

    這篇文章主要介紹了Java sleep方法及中斷方式、yield方法代碼實例,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-04-04
  • java8新特性-lambda表達式入門學(xué)習(xí)心得

    java8新特性-lambda表達式入門學(xué)習(xí)心得

    這篇文章主要介紹了java8新特性-lambda表達式入門學(xué)習(xí)心得,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-03-03
  • springboot項目如何部署到服務(wù)器

    springboot項目如何部署到服務(wù)器

    這篇文章主要介紹了springboot項目如何部署到服務(wù)器問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-05-05
  • Spring手寫簡化版MVC流程詳解

    Spring手寫簡化版MVC流程詳解

    Spring MVC是Spring Framework的一部分,是基于Java實現(xiàn)MVC的輕量級Web框架。本文將通過簡單示例帶大家掌握SpringMVC簡化版手寫方法,感興趣的可以了解一下
    2022-11-11
  • Java JDK8新增Optional工具類講解

    Java JDK8新增Optional工具類講解

    這篇文章主要介紹了Java JDK8新增Optional工具類講解,本文通過老版和jdk8對比對null的處理方式,講解了該項技術(shù)的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下
    2021-07-07
  • Java之MyBatis入門詳解

    Java之MyBatis入門詳解

    這篇文章主要介紹了Java之MyBatis入門詳解,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-12-12
  • SpringBoot項目啟動時預(yù)加載操作方法

    SpringBoot項目啟動時預(yù)加載操作方法

    Spring Boot是一種流行的Java開發(fā)框架,它提供了許多方便的功能來簡化應(yīng)用程序的開發(fā)和部署,這篇文章主要介紹了SpringBoot項目啟動時預(yù)加載,需要的朋友可以參考下
    2023-09-09
  • Lombok注解-@SneakyThrows的使用

    Lombok注解-@SneakyThrows的使用

    這篇文章主要介紹了Lombok注解-@SneakyThrows的使用,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-08-08
  • java中stream去重的幾種方式舉例

    java中stream去重的幾種方式舉例

    Stream流是數(shù)據(jù)渠道,用于操作數(shù)據(jù)源(集合、數(shù)組等)所生成的元素序列,這篇文章主要給大家介紹了關(guān)于java中stream去重的幾種方式,需要的朋友可以參考下
    2023-07-07

最新評論