欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java通過動(dòng)態(tài)代理實(shí)現(xiàn)一個(gè)簡(jiǎn)單的攔截器操作

 更新時(shí)間:2021年07月09日 11:09:03   作者:李白csdn  
這篇文章主要介紹了Java通過動(dòng)態(tài)代理實(shí)現(xiàn)一個(gè)簡(jiǎn)單的攔截器操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

一、代理

在使用動(dòng)態(tài)代理實(shí)現(xiàn)攔截器之前我們先簡(jiǎn)單了解一下什么Java的代理。

代理,顧名思義,就是不直接操作被代理(下面都用目標(biāo)對(duì)象稱呼,聽起來舒服一些)對(duì)象,而是通過一個(gè)代理對(duì)象去間接的使用目標(biāo)對(duì)象中的方法。代理分為兩種模式,一種是靜態(tài)代理,一種是動(dòng)態(tài)代理。接下來先寫一個(gè)靜態(tài)代理的例子。

無論是靜態(tài)代理還是動(dòng)態(tài)代理,目標(biāo)對(duì)象(target)都要實(shí)現(xiàn)一個(gè)接口(interface),注意,如果使用cglib提供的代理,則不必實(shí)現(xiàn)接口,而是通過子類去實(shí)現(xiàn),暫不討論該種方式。

(1)先定義一個(gè)接口

public interface IUserDao {
    void save();
}

(2)定義目標(biāo)對(duì)象(target)

public class UserDaoImpl implements IUserDao {
    public void save() {
        System.out.println("--------已經(jīng)保存數(shù)據(jù)---------");
    }
}

(3)定義代理對(duì)象

public class UserDaoProxy implements IUserDao {
 private IUserDao target;//將目標(biāo)對(duì)象放到代理對(duì)象中
 public UserDaoProxy(IUserDao target){
  this.target = target;
  }
 public void save() {
     System.out.println("------開始事務(wù)------");
            target.save();
     System.out.println("-------提交事務(wù)------");
    }
}

測(cè)試一下:

public class Test {
public static void main(String[] args){
    IUserDao userDao = new UserDaoImpl();
    UserDaoProxy proxy = new UserDaoProxy(userDao);
    proxy.save();//通過代理對(duì)象調(diào)用save方法
    }
}

輸出結(jié)果為:

------開始事務(wù)------
--------已經(jīng)保存數(shù)據(jù)---------

-------提交事務(wù)------

這種方式有一個(gè)問題,就是代理對(duì)象必須也要實(shí)現(xiàn)被代理對(duì)象所實(shí)現(xiàn)的同一個(gè)接口,這就出現(xiàn)了嚴(yán)重的耦合。所以,下面使用一種改進(jìn)的方式,即動(dòng)態(tài)代理(jdk代理)。

動(dòng)態(tài)代理方式也需要目標(biāo)對(duì)象(target)實(shí)現(xiàn)一個(gè)接口

(1)定義一個(gè)接口(IUserDao)

(2)定義一個(gè)目標(biāo)對(duì)象類(UserDaoImpl)

(3)創(chuàng)建動(dòng)態(tài)代理類

public class ProxyFactory {
    //維護(hù)一個(gè)目標(biāo)對(duì)象
    private Object target; 
    public ProxyFactory(Object target) {
        this.target = target;
    }
 
    //給目標(biāo)對(duì)象生成代理對(duì)象
    public Object getProxyInstance() {
        System.out.println("----target class---" + target.getClass());
        System.out.println("----target interfaces---" +
            target.getClass().getInterfaces());
 
        return Proxy.newProxyInstance(target.getClass().getClassLoader(),
            target.getClass().getInterfaces(),
            new InvocationHandler() {
                public Object invoke(Object proxy, Method method, Object[] args)
                    throws Throwable {
                    System.out.println("----開始事務(wù)2-----");
 
                    //執(zhí)行目標(biāo)對(duì)象方法
                    Object returnValue = method.invoke(target, args);
                    System.out.println("----提交事務(wù)2----");
 
                    return returnValue;
                }
            });
    }
}

測(cè)試一下:

public class Test {
    public static void main(String[] args) {
        //目標(biāo)對(duì)象
        IUserDao target = new UserDao();
        System.out.println(target.getClass());
 
        //給目標(biāo)對(duì)象創(chuàng)建代理對(duì)象
        IUserDao proxy = (IUserDao) new ProxyFactory(target).getProxyInstance();
        System.out.println("----proxy----:" + proxy.getClass());
        proxy.save();
        proxy.delete();
    }
}

輸出結(jié)果:

class com.jd.pattern.proxy.dynamicProxy.UserDao
----target class---class com.jd.pattern.proxy.dynamicProxy.UserDao
----target interfaces---[Ljava.lang.Class;@1fb3ebeb
----proxy----:class com.sun.proxy.$Proxy0
----開始事務(wù)2-----
-----保存完成------
----提交事務(wù)2----
----開始事務(wù)2-----
----刪除完成----

----提交事務(wù)2----

二、使用動(dòng)態(tài)代理實(shí)現(xiàn)一個(gè)簡(jiǎn)單的攔截器

既然是采用動(dòng)態(tài)代理的方式,那么肯定會(huì)有 接口、目標(biāo)類、代理類,再加一個(gè)攔截器

1、定義一個(gè)接口

public interface BusinessFacade {
    void doSomething();
}

2、定義一個(gè)目標(biāo)對(duì)象

public class BusinessClass implements BusinessFacade {
    public void doSomething() {
        System.out.println("在業(yè)務(wù)組件BusinessClass中調(diào)用doSomething方法");
    }
}

3、創(chuàng)建攔截器

public class InterceptorClass {
    public void before() {
        System.out.println("在InterceptorClass中調(diào)用方法:before()");
    }
 
    public void after() {
        System.out.println("在InterceptorClass中調(diào)用方法:after()");
    }
}

4、創(chuàng)建代理

public class DynamicProxyHandler {
    //聲明被代理對(duì)象
    private Object target;
    //創(chuàng)建攔截器
    InterceptorClass interceptor = new InterceptorClass();
    //動(dòng)態(tài)生成一個(gè)代理對(duì)象,并綁定被代理類和代理處理器
    public Object getProxyInstance(final Object target) {
        this.target = target;
        return Proxy.newProxyInstance(target.getClass().getClassLoader(),
            target.getClass().getInterfaces(),
            new InvocationHandler() {
                public Object invoke(Object proxy, Method method, Object[] args)
                    throws Throwable {
                    interceptor.before();
                    Object result = method.invoke(target, args);
                    interceptor.after();
                    return result;
                }
            });
    }
}

測(cè)試一下:

public class Test {
    public static void main(String[] args) {
        //創(chuàng)建動(dòng)態(tài)代理工具
        DynamicProxyHandler proxyHandler = new DynamicProxyHandler();
        //創(chuàng)建業(yè)務(wù)組件
        BusinessFacade target = new BusinessClass();
        //獲取代理對(duì)象
        BusinessFacade proxy = (BusinessFacade) proxyHandler.getProxyInstance(target);
        //通過代理對(duì)象調(diào)用目標(biāo)對(duì)象方法
        proxy.doSomething();
    }
}

輸出結(jié)果:

在InterceptorClass中調(diào)用方法:before()
在業(yè)務(wù)組件BusinessClass中調(diào)用doSomething方法
在InterceptorClass中調(diào)用方法:after()

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • Spring:如何使用枚舉參數(shù)

    Spring:如何使用枚舉參數(shù)

    這篇文章主要介紹了springboot枚舉類型傳遞的步驟,幫助大家更好的理解和學(xué)習(xí)使用springboot,感興趣的朋友可以了解下,希望能給你帶來幫助
    2021-08-08
  • IDEA運(yùn)行導(dǎo)入的javaweb項(xiàng)目tomcat正常,但是運(yùn)行失敗404問題

    IDEA運(yùn)行導(dǎo)入的javaweb項(xiàng)目tomcat正常,但是運(yùn)行失敗404問題

    這篇文章主要介紹了IDEA運(yùn)行導(dǎo)入的javaweb項(xiàng)目tomcat正常但是運(yùn)行失敗404問題,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-07-07
  • Spring MVC 4.1.3 + MyBatis零基礎(chǔ)搭建Web開發(fā)框架(注解模式)

    Spring MVC 4.1.3 + MyBatis零基礎(chǔ)搭建Web開發(fā)框架(注解模式)

    本篇文章主要介紹了Spring MVC 4.1.3 + MyBatis零基礎(chǔ)搭建Web開發(fā)框架(注解模式),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下。
    2017-03-03
  • 一步步教你如何使用Java實(shí)現(xiàn)WebSocket

    一步步教你如何使用Java實(shí)現(xiàn)WebSocket

    websocket協(xié)議是基于TCP的一種新的網(wǎng)絡(luò)協(xié)議,它實(shí)現(xiàn)了瀏覽器與服務(wù)器的全雙工通訊-允許服務(wù)器主動(dòng)發(fā)起信息個(gè)客戶端,websocket是一種持久協(xié)議,http是非持久協(xié)議,下面這篇文章主要給大家介紹了關(guān)于如何使用Java實(shí)現(xiàn)WebSocket的相關(guān)資料,需要的朋友可以參考下
    2023-05-05
  • springboot應(yīng)用服務(wù)啟動(dòng)事件的監(jiān)聽實(shí)現(xiàn)

    springboot應(yīng)用服務(wù)啟動(dòng)事件的監(jiān)聽實(shí)現(xiàn)

    本文主要介紹了springboot應(yīng)用服務(wù)啟動(dòng)事件的監(jiān)聽實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-04-04
  • Springboot項(xiàng)目的Mapper中增加一個(gè)新的sql語句

    Springboot項(xiàng)目的Mapper中增加一個(gè)新的sql語句

    本文主要介紹了Springboot項(xiàng)目的Mapper中增加一個(gè)新的sql語句,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2024-05-05
  • SpringBoot使用spring.factories加載默認(rèn)配置的實(shí)現(xiàn)代碼

    SpringBoot使用spring.factories加載默認(rèn)配置的實(shí)現(xiàn)代碼

    在日常開發(fā)過程中,發(fā)布一些產(chǎn)品或者框架時(shí),會(huì)遇到某些功能需要一些配置才能正常運(yùn)行,這時(shí)我們需要的提供默認(rèn)配置項(xiàng),同時(shí)用戶也能覆蓋進(jìn)行個(gè)性化
    2024-06-06
  • Spring三級(jí)緩存解決循環(huán)依賴

    Spring三級(jí)緩存解決循環(huán)依賴

    本文主要介紹了Spring三級(jí)緩存解決循環(huán)依賴,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-12-12
  • 基于Java語言MD5加密Base64轉(zhuǎn)換方法

    基于Java語言MD5加密Base64轉(zhuǎn)換方法

    這篇文章主要為大家詳細(xì)介紹了基于Java語言的MD5加密Base64轉(zhuǎn)換方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-09-09
  • Java中的Optional類用法詳細(xì)講解

    Java中的Optional類用法詳細(xì)講解

    這篇文章詳細(xì)介紹了Java中Optional類的使用,包括創(chuàng)建Optional對(duì)象,Optional對(duì)象的常用方法,如get、orElse、orElseGet、orElseThrow等,以及Optional類結(jié)合Lambda表達(dá)式和StreamAPI的使用,需要的朋友可以參考下
    2024-10-10

最新評(píng)論