Java动态代理一——动态类Proxy的使用

1.什么是动态代理?

答:动态代理可以提供对另一个对象的访问,同时隐藏实际对象的具体事实。代理一般会实现它所表示的实际对象的接口。代理可以访问实际对象,但是延迟实现实际对象的部分功能,实际对象实现系统的实际功能,代理对象对客户隐藏了实际对象。客户不知道它是与代理打交道还是与实际对象打交道。
2.为什么使用动态代理?

答:因为动态代理可以对请求进行任何处理

3.使用它有哪些好处?

答:因为动态代理可以对请求进行任何处理
4.哪些地方需要动态代理?

答:不允许直接访问某些类;对访问要做特殊处理等

 

目前Java开发包中包含了对动态代理的支持,但是其实现只支持对接口的的实现。 其实现主要通过java.lang.reflect.Proxy类和java.lang.reflect.InvocationHandler接口。 

Proxy类主要用来获取动态代理对象,InvocationHandler接口用来约束调用者实现

以下为模拟案例,通过动态代理实现在方法调用前后向控制台输出两句字符串

目录结构

Java动态代理一——动态类Proxy的使用_第1张图片


定义一个HelloWorld接口

复制代码
1  package  com.ljq.test;
2 
3    /**
4  * 定义一个HelloWorld接口
5 
6  @author  jiqinlin
7  *
8  */
9    public interface  HelloWorld {
10  public void  sayHelloWorld();
11  }
复制代码


类HelloWorldImpl是HelloWorld接口的实现

复制代码
1  package  com.ljq.test;
2 
3    /**
4  * 类HelloWorldImpl是HelloWorld接口的实现
5 
6  @author  jiqinlin
7  *
8  */
9    public class  HelloWorldImpl  implements  HelloWorld{
10 
11  public void  sayHelloWorld() {
12  System.out.println( " HelloWorld! " );
13  }
14 
15  }
复制代码

HelloWorldHandler是 InvocationHandler接口实现

复制代码
1  package  com.ljq.test;
2 
3    import  java.lang.reflect.InvocationHandler;
4    import  java.lang.reflect.Method;
5 
6    /**
7  * 实现在方法调用前后向控制台输出两句字符串
8 
9  @author  jiqinlin
10  *
11  */
12    public class  HelloWorldHandler  implements  InvocationHandler{
13  // 要代理的原始对象
14    private  Object obj;
15 
16  public  HelloWorldHandler(Object obj) {
17  super ();
18  this .obj  =  obj;
19  }
20 
21  /**
22  * 在代理实例上处理方法调用并返回结果
23 
24  @param  proxy 代理类
25  @param  method 被代理的方法
26  @param  args 该方法的参数数组
27  */
28  public  Object invoke(Object proxy, Method method, Object[] args)  throws  Throwable {
29  Object result  = null ;
30  // 调用之前
31     doBefore();
32  // 调用原始对象的方法
33  result = method.invoke(obj, args);
34  // 调用之后
35  doAfter();
36  return  result;
37  }
38 
39  private void  doBefore(){
40  System.out.println( " before method invoke " );
41  }
42 
43  private void  doAfter(){
44  System.out.println( " after method invoke " );
45  }
46 
47  }
复制代码

测试类

复制代码
package com.ljq.test;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;


public class  HelloWorldTest {

public static void  main(String[] args) {
HelloWorld helloWorld
= new  HelloWorldImpl();
InvocationHandler handler
= new  HelloWorldHandler(helloWorld);

// 创建动态代理对象
HelloWorld proxy = (HelloWorld)Proxy.newProxyInstance(
helloWorld.getClass().getClassLoader(), 
helloWorld.getClass().getInterfaces(), 
handler);
proxy.sayHelloWorld();
}
}
复制代码

运行结果为:

 

案例二

Calculator.java

复制代码
 1 import java.math.BigDecimal;
 2 
 3 
 4 public interface Calculator {
 5     
 6     //加法
 7     BigDecimal add(String a,String b);
 8     
 9     //减法
10     BigDecimal sub(String a,String b);
11     
12     //乘法
13     BigDecimal mul(String a,String b);
14     
15     //除法
16     BigDecimal div(String a,String b);
17 }
复制代码

 

SimpleCalculator.java

复制代码
 1 import java.math.BigDecimal;
 2 
 3 
 4 public class SimpleCalculator  implements Calculator{
 5 
 6     @Override
 7     public BigDecimal add(String a, String b) {
 8         
 9         BigDecimal n1=new BigDecimal(a);
10         BigDecimal n2=new BigDecimal(b);
11         
12         return n1.add(n2);
13     }
14 
15     @Override
16     public BigDecimal sub(String a, String b) {
17         BigDecimal n1=new BigDecimal(a);
18         BigDecimal n2=new BigDecimal(b);
19         
20         return n1.subtract(n2);
21     }
22 
23     @Override
24     public BigDecimal mul(String a, String b) {
25         BigDecimal n1=new BigDecimal(a);
26         BigDecimal n2=new BigDecimal(b);
27         
28         return n1.multiply(n2);
29     }
30 
31     @Override
32     public BigDecimal div(String a, String b) {
33         BigDecimal n1=new BigDecimal(a);
34         BigDecimal n2=new BigDecimal(b);
35         
36         return n1.divide(n2);
37     }
38 
39 }
复制代码

 

SimpleCalculatorLoggingProxy.java

复制代码
 1 import java.lang.reflect.InvocationHandler;
 2 import java.lang.reflect.Method;
 3 import java.lang.reflect.Proxy;
 4 
 5 
 6 public class SimpleCalculatorLoggingProxy {
 7 
 8     private Calculator target;
 9     
10     public SimpleCalculatorLoggingProxy(Calculator target){
11         this.target=target;
12     }
13     
14     public Calculator getProxy(){
15         Calculator proxy=null;
16         
17         //
18         ClassLoader loader=target.getClass().getClassLoader();
19         
20         Class[] interfaces=new Class[]{Calculator.class};
21         
22         InvocationHandler handle=new InvocationHandler() {
23             
24             @Override
25             public Object invoke(Object proxy, Method method, Object[] args)
26                     throws Throwable {
27                 System.out.println("正在执行"+method.getName()+"方法,参数为"+args[0]+","+args[1]);
28                 return method.invoke(target, args);
29             }
30         };
31         
32         proxy= (Calculator) Proxy.newProxyInstance(loader, interfaces, handle);
33         return proxy;
34     }
35     
36 }
复制代码

 

测试代码:

复制代码
1 Calculator target=new SimpleCalculator();
2 
3 Calculator proxy=new SimpleCalculatorLoggingProxy(target).getProxy();
4 
5 System.out.println(proxy.add("1", "2")); 
6 
7 System.out.println(proxy.div("3", "2")); 
复制代码

 

输出结果:

正在执行add方法,参数为1,2
3
正在执行div方法,参数为3,2
1.5

你可能感兴趣的:(设计模式,java)