Java通過動態代理實現一個簡單的攔截器操作
一、代理
在使用動態代理實現攔截器之前我們先簡單瞭解一下什麼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()
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。