您好,登錄后才能下訂單哦!
這篇文章主要介紹“Java動態代理模式怎么實現”,在日常操作中,相信很多人在Java動態代理模式怎么實現問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”Java動態代理模式怎么實現”的疑惑有所幫助!接下來,請跟著小編一起來學習吧!
1. Overview
Java在java.lang.reflect包下,定義了自己的代理。利用這個包下的類,我們可以在運行時動態地創建一個代理類,實現一個或多個接口。并將方法的調用轉發到你所指定的類。因為實際代理是在運行時創建的,所以稱為:動態代理。
Proxy:完全由java產生的,而且實現了完整的subject接口。
InvocationHandler:Proxy上的任何方法調用都會被傳入此類,InvocationHandler控制對RealSubject的訪問。
因為Java已經幫助我們創建了Proxy類,我們需要有辦法告訴Proxy類你要做什么,我們不能像以前一樣把代碼寫入到Proxy類中,因為Proxy類不是我們實現的。那么我們應該放在哪里?放在InvocationHandler類中,InvocationHandler類是響應代理的任何調用。我們可以吧InvocationHandler想成是代理收到方法調用后,請求做實際工作的對象。
2. java.lang.reflect.InvocationHandler
被代理實例所實現的一個接口,內部只有一個invoke()方法,簽名如下;
Java代碼
public Object invoke(Object proxy, Method method, Object[] args)
當代理的方法被調用的時候,代理就會把這個調用轉發給InvocationHandler,也就會調用它的invoke()方法。
3. java.lang.reflect.Proxy
提供用于創建動態代理類和實例的靜態方法,它還是由這些方法創建的所有動態代理類的超類,我們經常使用的靜態方式是:
Java代碼
newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h)
4. 示例:
情形:自己可以查看修改姓名性別,但是不能修改rate。他人可以查看姓名,性別以及修改rate,但是不能修改姓名性別。
4.1 定義一個接口:
Java代碼
public interface Person { String getName(); String getGender(); void setName(String name); void setGender(String gender); void setRate(int rate); int getRate(); }
4.2 定義實現Person接口類
Java代碼
public class PersonImpl implements Person { String name; String gender; String interests; int rate; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getGender() { return gender; } public void setGender(String gender) { this.gender = gender; } public String getInterests() { return interests; } public void setInterests(String interests) { this.interests = interests; } public int getRate() { return rate; } public void setRate(int rate) { this.rate = rate; }
4.3 定義OwnerInvocationHandler類,表示如果為本人,則可以進行修改查看姓名性別。
Java代碼
public class OwnerInvocationHandler implements InvocationHandler{ private Person personBean; public OwnerInvocationHandler(Person personBean){ this.personBean = personBean; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws IllegalAccessException { try { if(method.getName().startsWith("get")){//如果方法名為get,就調用person類內的get相應方法 return method.invoke(personBean, args); }else if(method.getName().equals("setRate")){ // 如果方法是setRate,則拋出異常 throw new IllegalAccessException("access deny"); }else if(method.getName().startsWith("set")){ //如果為set,就調用person類內的set相應方法 return method.invoke(personBean, args); }else { System.out.println("non method invoke"); } } catch (InvocationTargetException e) { e.printStackTrace(); } return null; } }
4.4 定義NonInvocationHandler類,表示如果不為本人,則可以進行查看姓名性別和修改rate。
Java代碼
public class NonInvocationHandler implements InvocationHandler{ // private Person person; public NonInvocationHandler(Person person){ this.person = person; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { if(method.getName().startsWith("setRate")){ return method.invoke(person, args); }else if (method.getName().startsWith("get")){ return method.invoke(person, args); } else { System.out.println("non method invoke"); return null; } } }
4.5 測試類MyDynamicProxy
Java代碼
public class MyDynamicProxy { public Person getOwnerPersonBeanProxy(Person person){ return (Person)Proxy.newProxyInstance(person.getClass().getClassLoader(), person.getClass().getInterfaces(), new OwnerInvocationHandler(person)); } public Person getNonPersonBeanProxy(Person person){ return (Person)Proxy.newProxyInstance(person.getClass().getClassLoader(), person.getClass().getInterfaces(), new NonInvocationHandler(person)); } public static void main(String[] args) { MyDynamicProxy mdp = new MyDynamicProxy(); mdp.test(); } public void test(){ // Person person = getPersonBeanFromDB1(); Person personProxy = getOwnerPersonBeanProxy(person); System.out.println(personProxy.getName()); try { personProxy.setRate(2); } catch (Exception e) { System.out.println("can not setRate"); } // Person person1 = getPersonBeanFromDB1(); Person personProxy2 = getNonPersonBeanProxy(person1); System.out.println(personProxy2.getName()); personProxy2.setRate(2); System.out.println(personProxy2.getRate()); } private Person getPersonBeanFromDB1(){ Person pb = new PersonImpl(); pb.setName("remy"); pb.setGender("girl"); pb.setRate(1); return pb; }
輸出結果:
Java代碼
remy can not setRate remy 2
到此,關于“Java動態代理模式怎么實現”的學習就結束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續學習更多相關知識,請繼續關注億速云網站,小編會繼續努力為大家帶來更多實用的文章!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。