溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

如何用Java實現(xiàn)動態(tài)代理

發(fā)布時間:2021-09-28 15:22:19 來源:億速云 閱讀:161 作者:iii 欄目:開發(fā)技術(shù)

本篇內(nèi)容介紹了“如何用Java實現(xiàn)動態(tài)代理”的有關(guān)知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠?qū)W有所成!

目錄
  • 前言

  • 靜態(tài)代理

  • 動態(tài)代理

  • CGLib實現(xiàn)動態(tài)代理

  • 總結(jié)

前言

動態(tài)代理在Java中有著廣泛的應用,比如Spring AOP、Hibernate數(shù)據(jù)查詢、測試框架的后端mock、RPC遠程調(diào)用、Java注解對象獲取、日志、用戶鑒權(quán)、全局性異常處理、性能監(jiān)控,甚至事務處理等。

代理,指的是使用代理對象代替對其它對象的訪問,簡單點說,你求職時找的中介就是代理,那么在Java中,代理如何體現(xiàn)呢?

靜態(tài)代理

我們首先需要知道,何為靜態(tài)代理?靜態(tài)代理指的是在編譯期就對目標對象的方法進行增強,例如:

public class TestDemo {

    interface EmailService {
        void sendEmail(String emailContent);
    }

    static class EmailServiceImpl implements EmailService{

        @Override
        public void sendEmail(String emailContent) {
            System.out.println("發(fā)送了一封郵件,內(nèi)容為:" + emailContent);
        }
    }

    public static void main(String[] args) {
        EmailService emailService = new EmailServiceImpl();
        emailService.sendEmail("hello");
    }
}

現(xiàn)在若是想在發(fā)送郵件之前獲取一下當前的時間,則可以使用代理類對發(fā)郵件的方法進行增強:

public class TestDemo {

    interface EmailService {
        void sendEmail(String emailContent);
    }

    static class EmailServiceImpl implements EmailService{

        @Override
        public void sendEmail(String emailContent) {
            System.out.println("發(fā)送了一封郵件,內(nèi)容為:" + emailContent);
        }
    }

    static class EmailProxy implements EmailService{

        private final EmailService emailService;

        public EmailProxy(EmailService emailService) {
            this.emailService = emailService;
        }

        @Override
        public void sendEmail(String emailContent) {
            System.out.println(LocalDateTime.now());
            emailService.sendEmail(emailContent);
        }
    }

    public static void main(String[] args) {
        EmailService emailProxy = new EmailProxy(new EmailServiceImpl());
        emailProxy.sendEmail("hello");
    }
}

靜態(tài)代理的缺點非常明顯,編寫麻煩,且可擴展性不強,而動態(tài)代理的出現(xiàn),將徹底解決這些問題。

動態(tài)代理

動態(tài)代理與靜態(tài)代理恰恰相反,動態(tài)代理是在運行期對目標對象的某個方法進行增強,比如仍然是發(fā)郵件的服務,使用動態(tài)代理,即可這樣實現(xiàn):

public class TestDemo {

    interface EmailService {
        void sendEmail(String emailContent);
    }

    static class EmailServiceImpl implements EmailService {

        @Override
        public void sendEmail(String emailContent) {
            System.out.println("發(fā)送了一封郵件,內(nèi)容為:" + emailContent);
        }
    }

    public static void main(String[] args) {
        EmailService emailService = new EmailServiceImpl();
        EmailService emailProxy = (EmailService) Proxy.newProxyInstance(EmailServiceImpl.class.getClassLoader(), EmailServiceImpl.class.getInterfaces(), new InvocationHandler() {
            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                System.out.println(LocalDateTime.now());
                Object result = method.invoke(emailService, args);
                return result;
            }
        });
        emailProxy.sendEmail("hello");
    }
}

使用JDK提供的Proxy和InvocationHandler類能夠輕松實現(xiàn)動態(tài)代理,但這種方式也是有局限性的,就是被增強的類必須實現(xiàn)了接口,因為Proxy的參數(shù)中需要接收類的接口信息。

CGLib實現(xiàn)動態(tài)代理

CGLib的出現(xiàn),打破了這一僵局,使用CGLib,能夠增強任意的對象方法,即使你沒有實現(xiàn)任何接口,因為它是通過繼承的方式進行增強的。

下面就來演示一下如何使用CGLib,首先引入依賴:

<dependency>
    <groupId>cglib</groupId>
    <artifactId>cglib</artifactId>
    <version>3.3.0</version>
</dependency>

實現(xiàn)如下:

public class TestDemo {

    static class EmailServiceImpl {
        public void sendEmail(String emailContent) {
            System.out.println("發(fā)送了一封郵件,內(nèi)容為:" + emailContent);
        }
    }

    public static void main(String[] args) {
        EmailServiceImpl emailService = new EmailServiceImpl();
        EmailServiceImpl emailProxy = (EmailServiceImpl) Enhancer.create(emailService.getClass(), new MethodInterceptor() {
            @Override
            public Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
                System.out.println(LocalDateTime.now());
                Object obj = methodProxy.invokeSuper(o, args);
                return obj;
            }
        });
        emailProxy.sendEmail("hello");
    }
}

它的寫法與JDK提供的方式類似,通過Enhancer類的create()方法即可增強一個對象,并傳入對象的Class對象和一個MethodInterceptor接口的實現(xiàn)類,并在intercept()方法中對原方法進行增強。

“如何用Java實現(xiàn)動態(tài)代理”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實用文章!

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI