溫馨提示×

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

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java動(dòng)態(tài)代理模式怎么實(shí)現(xiàn)

發(fā)布時(shí)間:2022-01-06 17:38:17 來(lái)源:億速云 閱讀:124 作者:iii 欄目:編程語(yǔ)言

這篇文章主要介紹“Java動(dòng)態(tài)代理模式怎么實(shí)現(xiàn)”,在日常操作中,相信很多人在Java動(dòng)態(tài)代理模式怎么實(shí)現(xiàn)問(wèn)題上存在疑惑,小編查閱了各式資料,整理出簡(jiǎn)單好用的操作方法,希望對(duì)大家解答”Java動(dòng)態(tài)代理模式怎么實(shí)現(xiàn)”的疑惑有所幫助!接下來(lái),請(qǐng)跟著小編一起來(lái)學(xué)習(xí)吧!

1. Overview

Java在java.lang.reflect包下,定義了自己的代理。利用這個(gè)包下的類,我們可以在運(yùn)行時(shí)動(dòng)態(tài)地創(chuàng)建一個(gè)代理類,實(shí)現(xiàn)一個(gè)或多個(gè)接口。并將方法的調(diào)用轉(zhuǎn)發(fā)到你所指定的類。因?yàn)閷?shí)際代理是在運(yùn)行時(shí)創(chuàng)建的,所以稱為:動(dòng)態(tài)代理。

Java動(dòng)態(tài)代理模式怎么實(shí)現(xiàn)

Proxy:完全由java產(chǎn)生的,而且實(shí)現(xiàn)了完整的subject接口。

InvocationHandler:Proxy上的任何方法調(diào)用都會(huì)被傳入此類,InvocationHandler控制對(duì)RealSubject的訪問(wèn)。

因?yàn)镴ava已經(jīng)幫助我們創(chuàng)建了Proxy類,我們需要有辦法告訴Proxy類你要做什么,我們不能像以前一樣把代碼寫(xiě)入到Proxy類中,因?yàn)镻roxy類不是我們實(shí)現(xiàn)的。那么我們應(yīng)該放在哪里?放在InvocationHandler類中,InvocationHandler類是響應(yīng)代理的任何調(diào)用。我們可以吧InvocationHandler想成是代理收到方法調(diào)用后,請(qǐng)求做實(shí)際工作的對(duì)象。

2. java.lang.reflect.InvocationHandler

被代理實(shí)例所實(shí)現(xiàn)的一個(gè)接口,內(nèi)部只有一個(gè)invoke()方法,簽名如下;

Java代碼

public Object invoke(Object proxy, Method method, Object[] args)

當(dāng)代理的方法被調(diào)用的時(shí)候,代理就會(huì)把這個(gè)調(diào)用轉(zhuǎn)發(fā)給InvocationHandler,也就會(huì)調(diào)用它的invoke()方法。

3. java.lang.reflect.Proxy

提供用于創(chuàng)建動(dòng)態(tài)代理類和實(shí)例的靜態(tài)方法,它還是由這些方法創(chuàng)建的所有動(dòng)態(tài)代理類的超類,我們經(jīng)常使用的靜態(tài)方式是:

Java代碼

newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h)

4. 示例:

情形:自己可以查看修改姓名性別,但是不能修改rate。他人可以查看姓名,性別以及修改rate,但是不能修改姓名性別。

4.1 定義一個(gè)接口:

Java代碼

public interface Person {                  String getName();                  String getGender();                  void setName(String name);                  void setGender(String gender);                  void setRate(int rate);                  int getRate();     }

4.2 定義實(shí)現(xiàn)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類,表示如果為本人,則可以進(jìn)行修改查看姓名性別。

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,就調(diào)用person類內(nèi)的get相應(yīng)方法                         return method.invoke(personBean, args);                 }else if(method.getName().equals("setRate")){ // 如果方法是setRate,則拋出異常                     throw new IllegalAccessException("access deny");                 }else if(method.getName().startsWith("set")){  //如果為set,就調(diào)用person類內(nèi)的set相應(yīng)方法                     return method.invoke(personBean, args);                 }else {                     System.out.println("non method invoke");                 }             } catch (InvocationTargetException e) {                 e.printStackTrace();             }             return null;                       }              }

4.4 定義NonInvocationHandler類,表示如果不為本人,則可以進(jìn)行查看姓名性別和修改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 測(cè)試類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;     }

輸出結(jié)果:

Java代碼

remy     can not setRate     remy     2

到此,關(guān)于“Java動(dòng)態(tài)代理模式怎么實(shí)現(xiàn)”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識(shí),請(qǐng)繼續(xù)關(guān)注億速云網(wǎng)站,小編會(huì)繼續(xù)努力為大家?guī)?lái)更多實(shí)用的文章!

向AI問(wèn)一下細(xì)節(jié)

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

AI