首頁 >Java >java教程 >Java怎麼透過動態代理實現一個簡單的攔截器操作

Java怎麼透過動態代理實現一個簡單的攔截器操作

王林
王林轉載
2023-04-30 20:55:05944瀏覽

一、代理

在使用動態代理實作攔截器之前我們先簡單了解一下什麼Java的代理程式。

代理,顧名思義,就是不直接操作被代理(下面都用目標對象稱呼,聽起來舒服一些)對象,而是透過一個代理對象去間接的使用目標對像中的方法。代理分為兩種模式,一種是靜態代理,一種是動態代理。接下來先寫一個靜態代理的例子。

無論是靜態代理還是動態代理,目標物件(target)都要實現一個介面(interface),注意,如果使用cglib提供的代理,則不必實現接口,而是透過子類去實現,暫不討論該種方式。

(1)先定義一個介面

public interface IUserDao {
    void save();
}

(2)定義目標物件(target)

public class UserDaoImpl implements IUserDao {
    public void save() {
        System.out.println("--------已经保存数据---------");
    }
}

(3)定義代理物件

public class UserDaoProxy implements IUserDao {
 private IUserDao target;//将目标对象放到代理对象中
 public UserDaoProxy(IUserDao target){
  this.target = target;
  }
 public void save() {
     System.out.println("------开始事务------");
            target.save();
     System.out.println("-------提交事务------");
    }
}

測試一下:

public class Test {
public static void main(String[] args){
    IUserDao userDao = new UserDaoImpl();
    UserDaoProxy proxy = new UserDaoProxy(userDao);
    proxy.save();//通过代理对象调用save方法
    }
}

輸出結果為:

------開始交易------
--------已經儲存數據---------

-------提交交易------

這種方式有一個問題,就是代理物件必須也要實現被代理物件所實現的同一個接口,這就出現了嚴重的耦合。所以,下面使用一種改進的方式,即動態代理(jdk代理)。

動態代理程式方式也需要目標物件(target)實作一個介面

(1)定義一個介面(IUserDao)

(2)定義一個目標物件類別(UserDaoImpl )

(3)建立動態代理類別

public class ProxyFactory {
    //维护一个目标对象
    private Object target; 
    public ProxyFactory(Object target) {
        this.target = target;
    }
 
    //给目标对象生成代理对象
    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("----开始事务2-----");
 
                    //执行目标对象方法
                    Object returnValue = method.invoke(target, args);
                    System.out.println("----提交事务2----");
 
                    return returnValue;
                }
            });
    }
}

測試一下:

public class Test {
    public static void main(String[] args) {
        //目标对象
        IUserDao target = new UserDao();
        System.out.println(target.getClass());
 
        //给目标对象创建代理对象
        IUserDao proxy = (IUserDao) new ProxyFactory(target).getProxyInstance();
        System.out.println("----proxy----:" + proxy.getClass());
        proxy.save();
        proxy.delete();
    }
}

輸出結果:

##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
----開始交易2-----
-----儲存完成--- ---
----提交交易2----
----開始交易2-----
----刪除完成----

#----提交交易2----

二、使用動態代理實現一個簡單的攔截器

#既然是採用動態代理的方式,那麼肯定會有介面、目標類、代理類,再加一個攔截器

1、定義一個介面

public interface BusinessFacade {
    void doSomething();
}

2、定義一個目標物件

public class BusinessClass implements BusinessFacade {
    public void doSomething() {
        System.out.println("在业务组件BusinessClass中调用doSomething方法");
    }
}

3、建立攔截器

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

4、建立代理

public class DynamicProxyHandler {
    //声明被代理对象
    private Object target;
    //创建拦截器
    InterceptorClass interceptor = new InterceptorClass();
    //动态生成一个代理对象,并绑定被代理类和代理处理器
    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;
                }
            });
    }
}

測試一下:

public class Test {
    public static void main(String[] args) {
        //创建动态代理工具
        DynamicProxyHandler proxyHandler = new DynamicProxyHandler();
        //创建业务组件
        BusinessFacade target = new BusinessClass();
        //获取代理对象
        BusinessFacade proxy = (BusinessFacade) proxyHandler.getProxyInstance(target);
        //通过代理对象调用目标对象方法
        proxy.doSomething();
    }
}
輸出結果:

在InterceptorClass中呼叫方法:before()

在業務元件BusinessClass中呼叫doSomething方法
在InterceptorClass中呼叫方法:after()

以上是Java怎麼透過動態代理實現一個簡單的攔截器操作的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:yisu.com。如有侵權,請聯絡admin@php.cn刪除