java之动态代理

Posted vincent-yuan

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了java之动态代理相关的知识,希望对你有一定的参考价值。

 

1.由于静态代理产生的代理类过多,造成代码量急剧增加,所以在实际开发中,通常使用动态代理来实现代理功能。

2.在java中实现动态代理的方式有很多。Jdk中通过接口来实现动态代理,如果要实现类的动态代理可以使用 cglib。目前也可以通过 javassit 来实现代理。

3.Jdk的动态代理实现是通过一个类和一个接口来实现的。

InvocationHandler 是代理实例 的调用处理程序 实现的接口。

Object invoke(Object proxy,

              Method method,

              Object[] args)

              throws Throwable

在代理实例上处理方法调用并返回结果。在与方法关联的代理实例上调用方法时,将在调用处理程序上调用此方法。

参数: 

proxy - 在其上调用方法的代理实例

method - 对应于在代理实例上调用的接口方法的 Method 实例。Method 对象的声明类将是在其中声明方法的接口,该接口可以是代理类赖以继承方法的代理接口的超接口。

args - 包含传入代理实例上方法调用的参数值的对象数组,如果接口方法不使用参数,则为 null。基本类型的参数被包装在适当基本包装器类(如 java.lang.Integer  java.lang.Boolean)的实例中。

返回: 

从代理实例的方法调用返回的值。如果接口方法的声明返回类型是基本类型,则此方法返回的值一定是相应基本包装对象类的实例;否则,它一定是可分配到声明返回类型的类型。如果此方法返回的值为 null 并且接口方法的返回类型是基本类型,则代理实例上的方法调用将抛出 NullPointerException。否则,如果此方法返回的值与上述接口方法的声明返回类型不兼容,则代理实例上的方法调用将抛出 ClassCastException

抛出: 

Throwable - 从代理实例上的方法调用抛出的异常。该异常的类型必须可以分配到在接口方法的 throws 子句中声明的任一异常类型或未经检查的异常类型 java.lang.RuntimeException  java.lang.Error。如果此方法抛出经过检查的异常,该异常不可分配到在接口方法的 throws 子句中声明的任一异常类型,代理实例的方法调用将抛出包含此方法曾抛出的异常的 UndeclaredThrowableException

 

Proxy 提供用于创建动态代理类和实例的静态方法,它还是由这些方法创建的所有动态代理类的超类。

public static Object newProxyInstance(ClassLoader loader,

                                      Class<?>[] interfaces,

                                      InvocationHandler h)

                               throws IllegalArgumentException

返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序。此方法相当于:

     Proxy.getProxyClass(loader, interfaces).

         getConstructor(new Class[] InvocationHandler.class ).

         newInstance(new Object[] handler );

 

Proxy.newProxyInstance 抛出 IllegalArgumentException,原因与 Proxy.getProxyClass 相同。

参数: 

loader - 定义代理类的类加载器

interfaces - 代理类要实现的接口列表

h - 指派方法调用的调用处理程序

返回: 

一个带有代理类的指定调用处理程序的代理实例,它由指定的类加载器定义,并实现指定的接口

抛出: 

IllegalArgumentException - 如果违反传递到 getProxyClass 的参数上的任何限制

NullPointerException - 如果 interfaces 数组参数或其任何元素为 null,或如果调用处理程序 null

4.通过动态代理实现租房

MyInvocationHandler.java

public class MyInvocationHandler implements InvocationHandler
    //目标对象 ---真实对象
    private Object target;
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable 
        fare();    
        //调用真实对象 执行其方法
        return method.invoke(target, args);
    
    private void fare()
        System.out.println("中介收费");
    
    public void setTarget(Object target) 
        this.target = target;
    

Rent.java

public interface Rent 
    //租房
    public void rent();

Host.java

public class Host implements Rent
    @Override
    public void rent() 
        System.out.println("========将房屋出租=======");
    

Client.java

public class Client 
    public static void main(String[] args) 
        Host host = new Host();
        MyInvocationHandler handler = new MyInvocationHandler();
        handler.setTarget(host);
        Rent proxy=(Rent)Proxy.newProxyInstance(Client.class.getClassLoader(),
                host.getClass().getInterfaces(), handler);
        proxy.rent();
    

5.封装为一个类

public class DynamicProxy implements InvocationHandler
    private Object target;
    public void setTarget(Object target) 
        this.target = target;
    
    public Object getProxy()
        return Proxy.newProxyInstance(this.getClass().getClassLoader(),
                target.getClass().getInterfaces(), 
                this);
    
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable 
        before();
        //result方法执行的返回值
        Object result = method.invoke(target, args);
        after();
        return result;
    
    private void before()
        System.out.println("在真实方法执行前 添加新的功能");
    
    private void after()
        System.out.println("在真实方法执行后 添加新的功能");
    

总结:动态代理解决了静态代理的缺点;一个类可以代理多个真实对象。

 

以上是关于java之动态代理的主要内容,如果未能解决你的问题,请参考以下文章

Java 设计模式之代理模式,Java 静态代理,Java 动态代理

Java代理之(jdk静态代理/jdk动态代理/cglib动态代理/aop/aspectj)

java反射机制应用之动态代理

浅谈-Java设计模式之动态代理

Java之动态代理简介

Java重要技术(27)动态代理之查看代理对象的类型信息