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

时间:2021-06-03 15:40:09

1.什么是动态代理?

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

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

3.使用它有哪些好处?

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

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

 

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

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

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

目录结构

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

<br/>

定义一个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 }

<br/>

类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();
}
}

运行结果为:

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

 
 

案例二

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