用 Java 实现拦截器 Interceptor 的拦截功能

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

大家好,又见面了,我是你们的朋友全栈君。

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

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

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

接下来,我们就用 Java 语言来实现拦截器Interceptor的拦截功能:

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

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

1

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

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

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/146068.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • java 四舍五入保留小数的几种方式

    java 四舍五入保留小数的几种方式引用处:https://blog.csdn.net/u014704879/article/details/41479399/https://blog.csdn.net/shruber/article/details/78413706https://www.cnblogs.com/Dhouse/p/7776780.html方式一:BigDecimaldoubletpD=6.1…

  • cmd命令 拷贝某文件夹及其子文件夹文件到其它文件夹

    cmd命令 拷贝某文件夹及其子文件夹文件到其它文件夹

  • python sendfile_sendfile详解(转)[通俗易懂]

    python sendfile_sendfile详解(转)[通俗易懂]在apache,nginx,lighttpd等web服务器当中,都有一项sendfile相关的配置,在一些网上的资料都有谈到sendfile会提升文件传输性能,那sendfile到底是什么呢?它的原理又是如何呢?在传统的文件传输里面(read/write方式),在实现上其实是比较复杂的,需要经过多次上下文的切换,我们看一下如下两行代码:read(file,tmp_buf,len);write(…

  • 光电编码器工作原理[通俗易懂]

    光电编码器工作原理[通俗易懂]光电编码器工作原理点击打开链接根据原理的不同又可分为:增量型、绝对型和混合式增量型。光电编码器的主要工作原理为光电转换,是一种通过光电转换将输出轴的机械几何位移量转换为脉冲或数字量的传感器。光电编码器主要由光栅盘和光电检测装置构成,在伺服系统中,光栅盘与电动机同轴致使电动机的旋转带动光栅盘的旋转,再经光电检测装置输出若干个脉冲信号,根据该信号的每秒脉冲数便可计算当前电动机的转速

  • 合成控制法原理

    合成控制法原理合成控制法反事实思维是**鲁宾反事实分析框架(Rubin’scounterfactualframework)**下进行思考,本质思想是为处理组(treatedgroup)找到一个合适的控制组(controlgroup)。合成控制法的思想是对于处理组,选择与处理组具有相似特征且未受到政策干预的控制组。通过对若干个控制区进行线性组合,构造一个近似处理组的合成控制区域(syntheticcontrolregion).对于未受政策影响的潜在控制组(donorpool),通过权重向量给控制组内每

  • moxa串口服务器网页版用户名密码,moxa串口服务器设置密码

    moxa串口服务器设置密码内容精选换一换登录Windows操作系统的弹性云服务器时,需使用密码方式登录。因此,用户需先根据创建弹性云服务器时使用的密钥文件,获取该弹性云服务器初始安装时系统生成的管理员密码(Administrator帐户或Cloudbase-init设置的帐户)。该密码为随机密码,安全性高,请放心使用。请根据您的个人需求,通过管理控制台或API方式获取Windo登录Windows…

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号