你有被代理過嗎?講講開源框架都在用的代理模式

大家好,我是老三。

這節我們來看一個非常重要的設計模式——代理模式,儘管我們工作中可能很少用到,但它是很多框架重要功能的基石,肘,我們開始吧。

引言

節假日的地鐵站,你是否見過有人掏出電腦,原地輸出,原來是羣裏被瘋狂@……

告訴自己不準哭

用戶提問題,客服@我們解決,可能很多開發同學都經歷過這樣的場景。

用戶找到客服,提出問題,客服又找到開發同學,讓開發同學去解決問題,開發同學解決完,最後反饋給客服,客服再反饋到用戶。

站在用戶的視角,感覺就是客服解決了這個問題,這其實就是一種代理。

用戶向客服提問題

我們以這個例子,來看看Java怎麼實現代理模式的吧。

Java的三種代理模式實現

代理模式的定義:

Provide a surrogate or placeholder for another object to control access to it.(爲其他對象提供一種代理以控制對這個對象的訪問。)

簡單說,就是設置一箇中間代理來控制訪問原目標對象,達到增強原對象的功能和簡化訪問方式的目的。

代理模式通用類圖

Java實現代理模式分爲兩類三種,兩類是靜態代理動態代理,動態代理又可以分爲JDK動態代理CGLIB動態代理

Java實現代理模式

靜態代理

靜態代理比較簡單,代理類需要實現和目標接口類一樣的接口。

Solver靜態代理類圖

  • 接口類:ISolver

    public interface ISolver {
        void solve();
    }
    
  • 目標類:Solver

    public class Solver implements ISolver {
        @Override
        public void solve() {
            System.out.println("瘋狂掉頭髮解決問題……");
        }
    }
    
  • 代理類:SolverProxy,代理類也要實現接口,並且還要維護一個目標對象。

    public class SolverProxy implements ISolver {
        //目標對象
        private ISolver target;
    
        public SolverProxy(ISolver target) {
            this.target = target;
        }
    
        @Override
        public void solve() {
            System.out.println("請問有什麼能幫到您?");
            target.solve();
            System.out.println("問題已經解決啦!");
        }
    }
    
  • 客戶端;Client

    public class Client {
        public static void main(String[] args) {
            //目標對象:程序員
            ISolver developer = new Solver();
            //代理:客服小姐姐
            SolverProxy csProxy = new SolverProxy(developer);
            //目標方法:解決問題
            csProxy.solve();
        }
    }
    
  • 運行結果

    請問有什麼能幫到您?
    瘋狂掉頭髮解決問題……
    問題已經解決啦!
    

我們看到,通過靜態代理,可以在不修改目標對象的前提下擴展目標對象的功能。

但是,它也有一些問題:

  • 冗餘:由於代理對象要實現與目標對象一致的接口,會產生過多的代理類。
  • 維護性不佳:一旦接口增加方法,目標對象與代理對象都要進行修改。

JDK動態代理

JDK動態代理利用了JDK反射機制,動態地在內存中構建代理對象,從而實現對目標對象的代理功能。

它主要用到了兩個反射類的API:

  • java.lang.reflect Proxy| static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces, InvocationHandler h):返回一個指定接口的代理類實例,該接口可以將方法調用指派到指定的調用處理程序
  • java.lang.reflect InvocationHandler|Object invoke(Object proxy, Method method, Object[] args) :在代理實例上調用目標方法,並返回結果。

我們來看看使用JDK動態代理之後的客服代理場景。

JDK動態代理類圖

  • 接口類:ISolver

    public interface ISolver {
        void solve();
    }
    
  • 目標類:Solver,目標類需要實現接口類。

    public class Solver implements ISolver {
        @Override
        public void solve() {
            System.out.println("瘋狂掉頭髮解決問題……");
        }
    }
    
  • 動態代理工廠:ProxyFactory,這裏的動態代理工廠,不需要實現接口,直接採用反射的方式生成一個目標對象的代理對象實例。

    ps:這裏用了一個匿名內部類的方法,還有一種方法,動態代理類實現InvocationHandler接口,大體上類似,就不再給出例子了。

    public class ProxyFactory {
    
        // 維護一個目標對象
        private Object target;
    
        public ProxyFactory(Object target) {
            this.target = target;
        }
    
        // 爲目標對象生成代理對象
        public Object getProxyInstance() {
            return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(),
                    new InvocationHandler() {
                        @Override
                        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                            System.out.println("請問有什麼可以幫到您?");
    
                            // 調用目標對象方法
                            Object returnValue = method.invoke(target, args);
    
                            System.out.println("問題已經解決啦!");
                            return null;
                        }
                    });
        }
    }
    
    
  • 客戶端:Client

    客戶端生成一個代理對象實例,通過代理對象調用目標對象方法的時候,就會進入invoke()方法,最後是通過反射的方式調用目標對象的方法。

    public class Client {
        public static void main(String[] args) {
            //目標對象:程序員
            ISolver developer = new Solver();
            //代理:客服小姐姐
            ISolver csProxy = (ISolver) new ProxyFactory(developer).getProxyInstance();
            //目標方法:解決問題
            csProxy.solve();
        }
    }
    
  • 運行結果:

    請問有什麼可以幫到您?
    瘋狂掉頭髮解決問題……
    問題已經解決啦!
    

我們簡單總結一下靜態代理和動態代理的主要區別:

靜態代理動態代理最主要區別

  • 靜態代理在編譯時就已經實現,編譯完成後代理類是一個實際的class文件
  • 動態代理是在運行時動態生成的,即編譯完成後沒有實際的class文件,而是在運行時動態生成類字節碼,並加載到JVM中

我們也觀察到,JDK動態代理,目標對象必須得實現接口,也就是說它是面向接口的,假如我們不想要接口怎麼辦呢?

Cglib動態代理

CGLIB(Code Generation Library)是一個基於ASM的字節碼生成庫,它允許我們在運行時對字節碼進行修改和動態生成,它是通過繼承來實現的。

我們來看看使用Cglib之後,我們的客服代理是什麼樣的:

Cglib動態代理類圖

  • 引入依賴:Cglib是第三方類庫,需要引入依賴

            <dependency>
                <groupId>cglib</groupId>
                <artifactId>cglib</artifactId>
                <version>3.2.5</version>
            </dependency>
    
  • 目標類:Solver,這裏目標類不用再實現接口。

    public class Solver {
    
        public void solve() {
            System.out.println("瘋狂掉頭髮解決問題……");
        }
    }
    
  • 動態代理工廠:

    public class ProxyFactory implements MethodInterceptor {
    
       //維護一個目標對象
        private Object target;
    
        public ProxyFactory(Object target) {
            this.target = target;
        }
    
        //爲目標對象生成代理對象
        public Object getProxyInstance() {
            //工具類
            Enhancer en = new Enhancer();
            //設置父類
            en.setSuperclass(target.getClass());
            //設置回調函數
            en.setCallback(this);
            //創建子類對象代理
            return en.create();
        }
    
        @Override
        public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
            System.out.println("請問有什麼可以幫到您?");
            // 執行目標對象的方法
            Object returnValue = method.invoke(target, args);
            System.out.println("問題已經解決啦!");
            return null;
        }
    
    }
    
  • 客戶端:Client

    public class Client {
        public static void main(String[] args) {
            //目標對象:程序員
            Solver developer = new Solver();
            //代理:客服小姐姐
            Solver csProxy = (Solver) new ProxyFactory(developer).getProxyInstance();
            //目標方法:解決問題
            csProxy.solve();
        }
    }
    
  • 運行結果

    請問有什麼可以幫到您?
    瘋狂掉頭髮解決問題……
    問題已經解決啦!
    

我們可以看到Cglib動態代理和JDK動態代理最大的區別就是:

  • 使用JDK動態代理的對象必須實現一個或多個接口
  • 使用Cglib動態代理的對象則無需實現接口,達到代理類無侵入。

我們還需要注意:

  • CGLib不能對聲明爲final的方法進行代理,因爲是通過繼承父類的方式實現,如果父類是final的,那麼就無法繼承父類。

擴展:動態代理的應用

標題裏說了,開源框架都在用的代理模式,那麼主流的開源框架哪些地方用到了代理模式呢?——確切說是動態代理呢?

比如:

  • Spring AOP可以將一些非核心業務流程抽象成切面,幫助我們處理非主線的流程,它是怎麼實現的呢?
  • 你平時用慣了的MyBatis,寫了那麼多Mapper接口,爲什麼它不用實現類就能執行SQL呢?

後面兩期是大家期待的面渣逆襲系列,將會揭曉這兩個問題的答案。


簡單的事情重複做,重複的事情認真做,認真的事情有創造性地做!

我是三分惡,一個能文能武的普通開發。點贊關注不迷路,咱們下期見!



參考:

[1].《設計模式之禪》

[2].Java三種代理模式:靜態代理、動態代理和cglib代理

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章