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

詳解Java JDK動態(tài)代理

 更新時間:2020年08月18日 11:46:29   作者:弗蘭克的貓  
這篇文章主要介紹了Java JDK動態(tài)代理的相關資料,幫助大家更好的理解和學習Java 代理的有關知識,感興趣的朋友可以了解下

  今天來看看Java的另一種代理方式——JDK動態(tài)代理

  我們之前所介紹的代理方式叫靜態(tài)代理,也就是靜態(tài)的生成代理對象,而動態(tài)代理則是在運行時創(chuàng)建代理對象。動態(tài)代理有更強大的攔截請求功能,因為可以獲得類的運行時信息,可以根據運行時信息來獲得更為強大的執(zhí)(騷)行(操)力(作)。

  我們還是以上一個例子為例,這里的IStars接口和Stars類都不需要修改,只需要修改代理類。

  創(chuàng)建JDK動態(tài)代理需要先實現InvocationHandler接口,并重寫其中的invoke方法,具體步驟如下:

  1. 創(chuàng)建一個類實現InvocationHandler接口。

  2. 給Proxy類提供委托類的ClassLoader和Interfaces來創(chuàng)建動態(tài)代理類。

  3. 利用反射機制得到動態(tài)代理類的構造函數。

  4. 利用動態(tài)代理類的構造函數創(chuàng)建動態(tài)代理類對象。

  我們用動態(tài)代理來改造一下之前的類:

  接口和委托類不需要修改:

public interface IStars {
 void sing();
 void dance();
}
public class Stars implements IStars{
 private String name;

 public Stars(String name) {
  this.name = name;
 }

 public String getName() {
  return name;
 }

 public void setName(String name) {
  this.name = name;
 }

 public void sing(){
  System.out.println(getName() + " 唱了一首歌.");
 }

 public void dance(){
  System.out.println(getName() + " 跳了一支舞.");
 }
}

  這是使用動態(tài)代理后的代理類:

public class StarsNewProxy implements InvocationHandler {

 //代理類持有委托類的對象引用
 private Object object;

 //保存sing和dance的次數
 private int num;

 public StarsNewProxy(Object object){
  this.object = object;
 }

 @Override
 public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  if (!runBefore(method)){
   return null;
  };
  //利用反射機制將請求分派給委托類處理,Method的invoke返回Object對象作為方法執(zhí)行結果
  Object result = method.invoke(object,args);
  runAfter(method);
  return result;
 }

 private boolean runBefore(Method method){
  System.out.println("我是代理,攔截到請求");
  if (method.getName().equals("dance")){
   System.out.println("抱歉,明星腳受傷了,不能跳舞表演了。");
   return false;
  }
  return true;
 }

 private void runAfter(Method method){
  System.out.println("我是代理,請求處理完畢");
 }
}

  新建一個工廠類來返回代理實例:

public class StarsNewProxyFactory {
 //構建工廠類,客戶類調用此方法獲得代理對象
 //對于客戶類而言,代理類對象和委托類對象是一樣的,不需要知道具體返回的類型
 public static IStars getInstance(String name){
  IStars stars = new Stars(name);
  InvocationHandler handler = new StarsNewProxy(stars);
  IStars proxy = null;
  proxy = (IStars) Proxy.newProxyInstance(
    stars.getClass().getClassLoader(),
    stars.getClass().getInterfaces(),
    handler
  );
  return proxy;
 }
}

  改寫一下測試類:

public class Test {
 public static void main(String[] args){
//  testA();
  testB();
 }

 /**
  * 靜態(tài)代理
  */
 private static void testA(){
  //創(chuàng)建目標對象
  IStars stars = new Stars("Frank");

  //代理對象,把目標傳給代理對象,建立關系
  IStars starsProxy = new StarsProxy(stars);
  for (int i = 0;i < 5; i++){
   starsProxy.sing();
  }
 }

 /**
  * JDK動態(tài)代理
  */
 private static void testB(){
  IStars proxy = StarsNewProxyFactory.getInstance("Frank");
  proxy.dance();
  proxy.sing();
 }
}

  輸出如下:

我是代理,攔截到請求
抱歉,明星腳受傷了,不能跳舞表演了。
我是代理,攔截到請求
Frank 唱了一首歌.
我是代理,請求處理完畢

  使用動態(tài)代理時實現了InvocationHandler接口并重寫了invoke方法,invoke方法的三個參數:

Object invoke(Object proxy, Method method, Object[] args) throws Throwable

proxy:  被代理的對象
method:  被代理對象的某個方法的Method對象
args:  被代理對象的某個方法接受的參數

  Proxy的newProxyInstance方法詳情如下:

public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) throws IllegalArgumentException

loader:  一個ClassLoader對象,定義了由哪個ClassLoader對象來對生成的代理對象進行加載

interfaces:  一個Interface對象的數組,表示的是我將要給我需要代理的對象提供一組什么接口,如果我提供了一組接口給它,那么這個代理對象就宣稱實現了該接口(多態(tài)),這樣我就能調用這組接口中的方法了

h:  一個InvocationHandler對象,表示的是當我這個動態(tài)代理對象在調用方法的時候,會關聯到哪一個InvocationHandler對象上

  可以看到,這里的動態(tài)代理跟靜態(tài)代理一樣,在代理類內部保存了一個委托類的實例,實際上都是調用原來的委托實例來進行需要的操作,代理類相當于給委托類加上一個外殼,把委托類置于代理類的內部,從而可以控制客戶類對委托類的訪問,就像上例中,代理類攔截了客戶類對Stars類的dance方法的訪問,并且輸出了補充信息。

  動態(tài)代理跟靜態(tài)代理最大的不同便是生成代理類的時期不同,靜態(tài)代理是在編譯期,而動態(tài)代理則是在運行時根據委托類信息動態(tài)生成。

  其次,動態(tài)代理實現的是InvocationHandler接口,而靜態(tài)代理則是直接實現公共接口。當然動態(tài)代理也是需要實現相同的接口的,只是將接口信息放在了getInstance內部,相當于代理類跟委托類之間的約定,“這幾個方法幫我代理一下吧”。

  最后,動態(tài)代理可以獲得更多的運行時信息,使用起來也會更加靈活。

  至此,JDK動態(tài)代理講解完畢,歡迎大家繼續(xù)關注!

以上就是詳解Java JDK動態(tài)代理的詳細內容,更多關于Java JDK動態(tài)代理的資料請關注腳本之家其它相關文章!

相關文章

  • Springboot項目實現Mysql多數據源切換的完整實例

    Springboot項目實現Mysql多數據源切換的完整實例

    這篇文章主要給大家介紹了關于Springboot項目實現Mysql多數據源切換的完整實例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-11-11
  • 深入理解可視化JVM 故障處理工具

    深入理解可視化JVM 故障處理工具

    這篇文章主要介紹了深入理解可視化JVM 故障處理工具,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-10-10
  • Java文件操作之按行讀取文件和遍歷目錄的方法

    Java文件操作之按行讀取文件和遍歷目錄的方法

    這篇文章主要介紹了Java文件操作之按行讀取文件和遞歸遍歷目錄的方法,遍歷目錄文中分別舉了遞歸和非遞歸的例子,需要的朋友可以參考下
    2016-03-03
  • 利用exe4j生成java的exe文件

    利用exe4j生成java的exe文件

    本文主要介紹了利用exe4j生成java的exe文件,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-09-09
  • java實現往hive 的map類型字段寫數據

    java實現往hive 的map類型字段寫數據

    這篇文章主要介紹了java實現往hive 的map類型字段寫數據操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-07-07
  • Java根據key獲取枚舉值的操作方法

    Java根據key獲取枚舉值的操作方法

    枚舉(enum)算一種“語法糖”,是指一個經過排序的、被打包成一個單一實體的項列表,一個枚舉的實例可以使用枚舉項列表中任意單一項的值,本文給大家介紹了Java?如何快速根據?key?獲取枚舉的值,需要的朋友可以參考下
    2024-07-07
  • restTemplate超時處理的策略方案

    restTemplate超時處理的策略方案

    這篇文章主要為大家介紹了restTemplate超時處理的策略方案,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-06-06
  • SpringBoot 配置文件加載位置與優(yōu)先級問題詳解

    SpringBoot 配置文件加載位置與優(yōu)先級問題詳解

    這篇文章主要介紹了SpringBoot 配置文件加載位置與優(yōu)先級問題,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-09-09
  • RocketMQ順序消息的原理與特點

    RocketMQ順序消息的原理與特點

    RocketMQ作為一款純java、分布式、隊列模型的開源消息中間件,支持事務消息、順序消息、批量消息、定時消息、消息回溯等,本篇我們了解如何實現順序消息的原理與特點
    2023-02-02
  • Java 壓縮圖片并打包成ZIP文件的示例

    Java 壓縮圖片并打包成ZIP文件的示例

    這篇文章主要介紹了Java 壓縮圖片并打包成ZIP文件的示例,幫助大家更好的理解和使用Java,感興趣的朋友可以了解下
    2020-12-12

最新評論