程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> 關於JAVA >> 代理模式之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代碼

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

當代理的方法被調用的時候,代理就會把這個調用轉發給InvocationHandler,也就會調用它的invoke()方法。

3. Java.lang.reflect.Proxy

提供用於創建動態代理類和實例的靜態方法,它還是由這些方法創建的所有動態代理類的超類,我們經常使用的靜態方式是:

Java代碼

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

4. 示例:

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

4.1 定義一個接口:

Java代碼

  1. public interface Person {
  2. String getName();
  3. String getGender();
  4. void setName(String name);
  5. void setGender(String gender);
  6. void setRate(int rate);
  7. int getRate();
  8. }

4.2 定義實現Person接口類

Java代碼

  1. public class PersonImpl implements Person {
  2. String name;
  3. String gender;
  4. String interests;
  5. int rate;
  6. public String getName() {
  7. return name;
  8. }
  9. public void setName(String name) {
  10. this.name = name;
  11. }
  12. public String getGender() {
  13. return gender;
  14. }
  15. public void setGender(String gender) {
  16. this.gender = gender;
  17. }
  18. public String getInterests() {
  19. return interests;
  20. }
  21. public void setInterests(String interests) {
  22. this.interests = interests;
  23. }
  24. public int getRate() {
  25. return rate;
  26. }
  27. public void setRate(int rate) {
  28. this.rate = rate;
  29. }

4.3 定義OwnerInvocationHandler類,表示如果為本人,則可以進行修改查看姓名性別。

Java代碼

  1. public class OwnerInvocationHandler implements InvocationHandler{
  2. private Person personBean;
  3. public OwnerInvocationHandler(Person personBean){
  4. this.personBean = personBean;
  5. }
  6. @Override
  7. public Object invoke(Object proxy, Method method, Object[] args)
  8. throws IllegalAccessException {
  9. try {
  10. if(method.getName().startsWith("get")){//如果方法名為get,就調用person類內的get相應方法
  11. return method.invoke(personBean, args);
  12. }else if(method.getName().equals("setRate")){ // 如果方法是setRate,則拋出異常
  13. throw new IllegalAccessException("Access deny");
  14. }else if(method.getName().startsWith("set")){ //如果為set,就調用person類內的set相應方法
  15. return method.invoke(personBean, args);
  16. }else {
  17. System.out.println("non method invoke");
  18. }
  19. } catch (InvocationTargetException e) {
  20. e.printStackTrace();
  21. }
  22. return null;
  23. }
  24. }

4.4 定義NonInvocationHandler類,表示如果不為本人,則可以進行查看姓名性別和修改rate。

Java代碼

  1. public class NonInvocationHandler implements InvocationHandler{
  2. //
  3. private Person person;
  4. public NonInvocationHandler(Person person){
  5. this.person = person;
  6. }
  7. @Override
  8. public Object invoke(Object proxy, Method method, Object[] args)
  9. throws Throwable {
  10. if(method.getName().startsWith("setRate")){
  11. return method.invoke(person, args);
  12. }else if (method.getName().startsWith("get")){
  13. return method.invoke(person, args);
  14. } else {
  15. System.out.println("non method invoke");
  16. return null;
  17. }
  18. }
  19. }

4.5 測試類MyDynamicProxy

Java代碼

  1. public class MyDynamicProxy {
  2. public Person getOwnerPersonBeanProxy(Person person){
  3. return (Person)Proxy.newProxyInstance(person.getClass().getClassLoader(),
  4. person.getClass().getInterfaces(), new OwnerInvocationHandler(person));
  5. }
  6. public Person getNonPersonBeanProxy(Person person){
  7. return (Person)Proxy.newProxyInstance(person.getClass().getClassLoader(),
  8. person.getClass().getInterfaces(), new NonInvocationHandler(person));
  9. }
  10. public static void main(String[] args) {
  11. MyDynamicProxy mdp = new MyDynamicProxy();
  12. mdp.test();
  13. }
  14. public void test(){
  15. //
  16. Person person = getPersonBeanFromDB1();
  17. Person personProxy = getOwnerPersonBeanProxy(person);
  18. System.out.println(personProxy.getName());
  19. try {
  20. personProxy.setRate(2);
  21. } catch (Exception e) {
  22. System.out.println("can not setRate");
  23. }
  24. //
  25. Person person1 = getPersonBeanFromDB1();
  26. Person personProxy2 = getNonPersonBeanProxy(person1);
  27. System.out.println(personProxy2.getName());
  28. personProxy2.setRate(2);
  29. System.out.println(personProxy2.getRate());
  30. }
  31. private Person getPersonBeanFromDB1(){
  32. Person pb = new PersonImpl();
  33. pb.setName("remy");
  34. pb.setGender("girl");
  35. pb.setRate(1);
  36. return pb;
  37. }

輸出結果:

Java代碼

  1. remy
  2. can not setRate
  3. remy
  4. 2
  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved