动态代理拦截器示例

Java 里的拦截器是动态拦截 action 调用的对象,它提供了一种机制可以使开发者可以定义在一个 action 执行的前后执行的代码,也可以在一个 action 执行前阻止其执行,同时也提供了一种可以提取 action 中可重用部分的方式。在 AOP(Aspect-Oriented Programming)中拦截器用于在某个方法或字段被访问之前进行拦截,然后在之前或之后加入某些操作。

此外,拦截器在流行的开源框架中也很常见,其依赖的技术就是 Java 的动态代理。理解拦截器的核心原理对理解这些开源框架的体系结构至关重要。下面,我们就以一个简单的模型的来说明拦截器实现的一般方法。模型主要分为五个模块,分别:

业务组件,被代理和被拦截的对象;
代理处理器,实现了InvocationHandler接口的一个对象;
代理对象,Proxy对象;
拦截器,普通的 Java Bean,在调用业务方法之前或者之后会自动拦截并执行自己的一些方法;
客户端,执行业务处理的入口。

第 1 步:

创建业务组件接口 BusinessFacade

/**
 * @author 维C果糖
 * @create 2017-03-30
 *
 * GitHub:github.com/guobinhit
 *
 * 业务组件接口
 */
public interface BusinessFacade {
    public void doSomething();
}

第 2步:

创建业务组件实现类 BusinessClass

/**
 * @author 维C果糖
 * @create 2017-03-30
 *
 * GitHub:github.com/guobinhit
 *
 * 业务组件接口的实现类
 */
public class BusinessClass implements BusinessFacade {
    public void doSomething() {
        System.out.println("在业务组件 BusinessClass 中调用方法: doSomething()");
    }
}

第 3 步:

创建拦截器 InterceptorClass
/**

  • @author 维C果糖

  • @create 2017-03-30

  • GitHub:github.com/guobinhit

  • 拦截器
    */
    public class InterceptorClass {
    // 在 action 之前调用
    public void before(){
    System.out.println(“在拦截器 InterceptorClass 中调用方法: before()”);
    }

    // 在 action 之后调用
    public void after(){
    System.out.println(“在拦截器 InterceptorClass 中调用方法: after()”);
    }
    }

第 4 步:

创建动态代理处理器工具 DynamicProxyHandler

/**
 * @author 维C果糖
 * @create 2017-03-30
 *
 * GitHub:github.com/guobinhit
 *
 * 动态代理处理器工具
 */
public class DynamicProxyHandler implements InvocationHandler {
    // 声明被代理对象
    private Object business;

    // 创建拦截器
    private InterceptorClass interceptor = new InterceptorClass();

    /**
   * 动态生成一个代理类对象,并绑定被代理类和代理处理器。
   *
   * @param business
   * @return 代理类对象
   */
    public Object bind(Object business) {
        this.business = business;

        /**
         * Proxy.newProxyInstance(参数1, 参数2, 参数3)
         *
         * 参数1, 表示被代理类的 ClassLoader
         * 参数2, 表示被代理的接口
         * 参数3, 表示代理处理器对象
         *
         * 该方法,返回代理实例
         */
        return Proxy.newProxyInstance(business.getClass().getClassLoader(),
                business.getClass().getInterfaces(),
                this);
    }

    /**
   * 代理需要调用的方法,并在该方法调用前后,先调用连接器的方法。
   *
   * @param proxy 代理类对象
   * @param method 被代理的接口方法
   * @param args 被代理接口方法的参数
   * @return 方法调用返回的结果
   * @throws Throwable
   */
   public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
       Object result = null;
       interceptor.before();
       result = method.invoke(business, args);
       interceptor.after();
       return null;
   }
}

第 5 步:

创建 ClientDemo

/**
 * @author 维C果糖
 * @create 2017-03-30
 *
 * GitHub:github.com/guobinhit
 *
 * 客户端
 */
public class ClientDemo {
    public static void main(String args[]) {
        // 创建动态代理处理工具
        DynamicProxyHandler handler = new DynamicProxyHandler();

        // 创建业务组件对象
        BusinessFacade business = new BusinessClass();

        // 创建业务组件对象,并用动态代理绑定代理类
        BusinessFacade businessProxy = (BusinessFacade) handler.bind(business);

        // 调用业务组件中的方法,演示拦截器效果
        businessProxy.doSomething();
    }
}

运行上面的项目代码,结果如下图所示:

在这里插入图片描述

如上图所示,显然我们拦截器的拦截功能实现啦!

通过这篇文章,我们可能会对拦截器的实现原理有一个更透彻的理解。But,在真正的项目实践之中,要想实现拦截器的功能,我们一般采用继承类HandlerInterceptorAdapter或者抽象类AbstractInterceptor,或者实现HandleInterceptor接口。也就是说,我们只需要关心如何重写方法,而不需要关心其内部的实现原理。

原文链接:https://blog.csdn.net/qq_35246620/article/details/68484407

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值