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

java面試常見模式問題---代理模式

 更新時間:2021年06月09日 14:07:15   作者:興趣使然の草帽路飛  
代理模式是常用的java設計模式,他的特征是代理類與委托類有同樣的接口,代理類主要負責為委托類預處理消息、過濾消息、把消息轉(zhuǎn)發(fā)給委托類,以及事后處理消息
  • 本篇總結(jié)的是 代理設計模式,后續(xù)會經(jīng)常更新~
  • 代理模式最直觀的解釋就是,通過代理,將被代理對象 “增強”!(即,擴展被代理對象的功能)
  • 代理模式分為靜態(tài)代理,和動態(tài)代理:動態(tài)代理的代理類是動態(tài)生成的 , 靜態(tài)代理的代理類是我們提前寫好的邏輯。
  • Java 中實現(xiàn)動態(tài)代理的方式有 2 種:
  • JDK 動態(tài)代理
  • CGLIB 動態(tài)代理

1、靜態(tài)代理

靜態(tài)代理角色分析

  • 抽象角色 :一般使用接口或者抽象類來實現(xiàn)。
  • 真實角色 :被代理的角色。
  • 代理角色: 代理真實角色 , 代理真實角色后 ,一般會做一些附屬的操作。
  • 調(diào)用方:使用代理角色來進行一些操作。

我們以租客租客租房子為例,涉及到的對象有:租客、中介、房東。(房東即為被代理對象,中介即為代理對象)

租客通過中介之手租住房東的房子,代理對象中介需要尋找租客租房,并從中獲取中介費用。

代碼實現(xiàn)

Rent.java 即抽象角色

// 抽象角色:租房
public interface Rent {
   public void rent();
}

Host.java 即真實角色

// 真實角色: 房東,房東要出租房子
public class Host implements Rent{
   public void rent() {
       System.out.println("房屋出租");
  }
}

Proxy.java 即代理角色

//代理角色:中介
public class Proxy implements Rent {
   private Host host;
   public Proxy() { }
   public Proxy(Host host) {
       this.host = host;
  }
   // 租房
   public void rent(){
       seeHouse();
       host.rent();
       fare();
  }
   // 看房
   public void seeHouse(){
       System.out.println("帶房客看房");
  }
   // 收中介費
   public void fare(){
       System.out.println("收中介費");
  }
}

Client.java 調(diào)用方,即客戶

// 客戶類,一般客戶都會去找代理!
public class Client {
   public static void main(String[] args) {
       // 房東要租房
       Host host = new Host();
       // 中介幫助房東
       Proxy proxy = new Proxy(host);
       // 你去找中介!
       proxy.rent();
  }
}

靜態(tài)代理的缺點

需要手動創(chuàng)建代理類,如果需要代理的對象多了,那么代理類也越來越多。

為了解決,這個問題,就有了動態(tài)代理 !

2、動態(tài)代理

說到動態(tài)代理,面試的時候肯定會問動態(tài)代理的兩種實現(xiàn)方式:

先來看公共的 UserService 接口,和 UserServiceImpl 實現(xiàn)類:

/**
 * @author csp
 * @date 2021-06-03
 */
public interface UserService {
    /**
     * 登錄
     */
    void login();
    /**
     * 登出
     */
    void logout();
}
/**
 * @author csp
 * @date 2021-06-03
 */
public class UserServiceImpl implements UserService{
    @Override
    public void login() {
        System.out.println("用戶登錄...");
    }
    @Override
    public void logout() {
        System.out.println("用戶推出登錄...");
    }
}

JDK 動態(tài)代理

代碼如下

/**
 * @author csp
 * @date 2021-06-03
 */
public class JDKProxyFactory implements InvocationHandler {
    // 目標對象(被代理對象)
    private Object target;
    public JDKProxyFactory(Object target) {
        super();
        this.target = target;
    }
    /**
     * 創(chuàng)建代理對象
     *
     * @return
     */
    public Object createProxy() {
        // 1.得到目標對象的類加載器
        ClassLoader classLoader = target.getClass().getClassLoader();
        // 2.得到目標對象的實現(xiàn)接口
        Class<?>[] interfaces = target.getClass().getInterfaces();
        // 3.第三個參數(shù)需要一個實現(xiàn)invocationHandler接口的對象
        Object newProxyInstance = Proxy.newProxyInstance(classLoader, interfaces, this);
        return newProxyInstance;
    }
    /**
     * 真正執(zhí)行代理增強的方法
     *
     * @param proxy  代理對象.一般不使用
     * @param method 需要增強的方法
     * @param args   方法中的參數(shù)
     * @return
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("JDK 動態(tài)代理:登錄/登出前邏輯校驗......");
        Object invoke = method.invoke(target, args);
        System.out.println("JDK 動態(tài)代理:登錄/登出后日志打印......");
        return invoke;
    }
    public static void main(String[] args) {
        // 1.創(chuàng)建對象
        UserServiceImpl userService = new UserServiceImpl();
        // 2.創(chuàng)建代理對象
        JDKProxyFactory jdkProxyFactory = new JDKProxyFactory(userService);
        // 3.調(diào)用代理對象的增強方法,得到增強后的對象
        UserService userServiceProxy = (UserService) jdkProxyFactory.createProxy();
        userServiceProxy.login();
        System.out.println("==================================");
        userServiceProxy.logout();
    }
}

輸出結(jié)果如下

JDK 動態(tài)代理:登錄/登出前邏輯校驗......
用戶登錄...
JDK 動態(tài)代理:登錄/登出后日志打印......
==================================
JDK 動態(tài)代理:登錄/登出前邏輯校驗......
用戶推出登錄...
JDK 動態(tài)代理:登錄/登出后日志打印......

CGLIB 動態(tài)代理

代碼如下:

/**
 * @author csp
 * @date 2021-06-03
 */
public class CglibProxyFactory implements MethodInterceptor {
    // 目標對象(被代理對象)
    private Object target;
    // 使用構(gòu)造方法傳遞目標對象
    public CglibProxyFactory(Object target) {
        super();
        this.target = target;
    }
    /**
     * 創(chuàng)建代理對象
     *
     * @return
     */
    public Object createProxy() {
        // 1.創(chuàng)建Enhancer
        Enhancer enhancer = new Enhancer();
        // 2.傳遞目標對象的class
        enhancer.setSuperclass(target.getClass());
        // 3.設置回調(diào)操作
        enhancer.setCallback(this);
        return enhancer.create();
    }
    /**
     * 真正執(zhí)行代理增強的方法
     * @param o 代理對象
     * @param method 要增強的方法
     * @param objects 要增強方法的參數(shù)
     * @param methodProxy 要增強的方法的代理
     * @return
     * @throws Throwable
     */
    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        System.out.println("cglib 動態(tài)代理:登錄/登出前邏輯校驗......");
        Object invoke = method.invoke(target, objects);
        System.out.println("cglib 動態(tài)代理:登錄/登出后日志打印......");
        return invoke;
    }
    public static void main(String[] args) {
        // 1.創(chuàng)建對象
        UserServiceImpl userService = new UserServiceImpl();
        // 2.創(chuàng)建代理對象
        CglibProxyFactory cglibProxyFactory = new CglibProxyFactory(userService);
        // 3.調(diào)用代理對象的增強方法,得到增強后的對象
        UserService userServiceProxy = (UserService) cglibProxyFactory.createProxy();
        userServiceProxy.login();
        System.out.println("==================================");
        userServiceProxy.logout();
    }
}

測試結(jié)果如下

cglib 動態(tài)代理:登錄/登出前邏輯校驗......
用戶登錄...
cglib 動態(tài)代理:登錄/登出后日志打印......
==================================
cglib 動態(tài)代理:登錄/登出前邏輯校驗......
用戶推出登錄...
cglib 動態(tài)代理:登錄/登出后日志打印......

面試題一:JDK動態(tài)代理和CGLIB動態(tài)代理區(qū)別?

① JDK 動態(tài)代理本質(zhì)上是實現(xiàn)了被代理對象的接口,而 CGLib 本質(zhì)上是繼承了被代理對象,覆蓋其中的方法。

② JDK 動態(tài)代理只能對實現(xiàn)了接口的類生成代理,CGLib 則沒有這個限制。但是 CGLib 因為使用繼承實現(xiàn),所以 CGLib 所以無法對 final private 方法static方法進行代理。

③ JDK 動態(tài)代理是 JDK 里自帶的,CGLib 動態(tài)代理需要引入第三方的 jar 包。

④ 在調(diào)用代理方法上,JDK動態(tài)代理是通過反射機制調(diào)用,CGLib 是通過 FastClass 機制直接調(diào)用。(看過一篇文章,介紹說 FastClass 簡單的理解,就是使用一個 index 下標作為入?yún)?,可以直接定位到要調(diào)用的方法直接,并進行調(diào)用)

在性能上,JDK1.7 之前,由于使用了 FastClass 機制,CGLib 在執(zhí)行效率上比 JDK 快,但是隨著 JDK 動態(tài)代理的不斷優(yōu)化,從 JDK 1.7 開始,JDK 動態(tài)代理已經(jīng)明顯比 CGLib 更快了。

面試題二:JDK 動態(tài)代理為什么只能對實現(xiàn)了接口的類生成代理?

根本原因是通過 JDK 動態(tài)代理生成的類已經(jīng)繼承了 Proxy 類,所以無法再使用繼承的方式去對類實現(xiàn)代理。

總結(jié)

文章會不定時更新,有時候一天多更新幾篇,如果幫助您復習鞏固了知識點,還請三連支持一下,后續(xù)會一點點的更新!希望大家多多關注腳本之家的其他內(nèi)容!

  • Java中volatile關鍵字的作用與用法詳解

    Java中volatile關鍵字的作用與用法詳解

    volatile關鍵字雖然從字面上理解起來比較簡單,但是要用好不是一件容易的事情。這篇文章主要介紹了Java中volatile關鍵字的作用與用法詳解的相關資料,需要的朋友可以參考下
    2016-09-09
  • JDK更換IDEA如何修改圖文詳解

    JDK更換IDEA如何修改圖文詳解

    這篇文章主要給大家介紹了關于JDK更換IDEA如何修改的相關資料,Java的不同版本的JDK之間存在細微的差別,文中通過圖文介紹的非常詳細,需要的朋友可以參考下
    2023-07-07
  • java編程題之順時針打印矩陣

    java編程題之順時針打印矩陣

    這篇文章主要為大家詳細介紹了java編程題之順時針打印矩陣,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-03-03
  • SpringBoot2零基礎到精通之數(shù)據(jù)與頁面響應

    SpringBoot2零基礎到精通之數(shù)據(jù)與頁面響應

    SpringBoot是一種整合Spring技術棧的方式(或者說是框架),同時也是簡化Spring的一種快速開發(fā)的腳手架
    2022-03-03
  • java8新特性之方法引用示例代碼

    java8新特性之方法引用示例代碼

    這篇文章主要給大家介紹了關于java8新特性之方法引用的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-11-11
  • 舉例分析Python中設計模式之外觀模式的運用

    舉例分析Python中設計模式之外觀模式的運用

    這篇文章主要介紹了Python中設計模式之外觀模式的運用,外觀模式主張以分多模塊進行代碼管理而減少耦合,需要的朋友可以參考下
    2016-03-03
  • Java中FileWriter類的常用方法說明

    Java中FileWriter類的常用方法說明

    這篇文章主要介紹了Java中FileWriter類的常用方法說明,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-07-07
  • Java的AQS基本原理詳細分析

    Java的AQS基本原理詳細分析

    這篇文章主要介紹了Java的AQS基本原理詳細分析,AQS是Abstract Queued Synchronizer的簡稱,AQS提供了一種實現(xiàn)阻塞鎖和一系列依賴FIFO等待隊列的同步器的框架,本文主要講解分析其基本原理,需要的朋友可以參考下
    2024-01-01
  • 詳解Java springboot 整合Shiro框架

    詳解Java springboot 整合Shiro框架

    這篇文章主要為大家介紹了Java springboot 整合Shiro框架,具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助
    2022-01-01
  • 最新評論