Java實現的攔截器

ysuncn發表於2020-04-05
攔截器在在流行的開源框架中很常見,依賴的技術就是Java的動態代理。
理解攔截器的核心原理對理解這些開源框架的體系結構至關重要。
下面以一個簡單的模型的來說明攔截器的實現的一般方法。
 
模型分為以下模組:
業務元件:是被代理和被攔截的物件。
代理處理器:實現了InvocationHandler介面的一個物件
代理物件:Proxy物件。
攔截器:普通的JavaBean,在呼叫業務方法的之前或者之後會自動攔截並執行自己的一些方法。
客戶端:執行業務處理的入口。
 
 
以下是模型的實現

一、業務元件:分為業務介面和業務類
/**
* Created by IntelliJ IDEA.
* User: leizhimin
* Date: 2008-3-20 23:32:06
* Company: LavaSoft(http://lavasoft.blog.51cto.com/)
* 業務元件介面
*/

public interface BusinessInterface {
    public void doSomething();
}
 
/**
* Created by IntelliJ IDEA.
* User: leizhimin
* Date: 2008-3-20 23:31:12
* Company: LavaSoft(http://lavasoft.blog.51cto.com/)
* 業務元件
*/

public class BusinessClass implements BusinessInterface{

    public void doSomething() {
        System.out.println("業務元件BusinessClass方法呼叫:doSomething()");
    }
}
 
 
二、代理處理器:包含了業務物件繫結動態代理類的處理,並實現了InvocationHandler介面的invoke方法。
 
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

/**
* Created by IntelliJ IDEA.
* User: leizhimin
* Date: 2008-3-20 23:24:10
* Company: LavaSoft(http://lavasoft.blog.51cto.com/)
* 動態代理處理器工具
*/

public class DynamicProxyHandler implements InvocationHandler {
    private Object business;    //被代理物件
    private InterceptorClass interceptor = new InterceptorClass();    //攔截器

    /**
     * 動態生成一個代理類物件,並繫結被代理類和代理處理器
     *
     * @param business
     * @return 代理類物件
     */

    public Object bind(Object business) {
        this.business = business;
        return Proxy.newProxyInstance(
                //被代理類的ClassLoader
                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;  //To change body of implemented methods use File | Settings | File Templates.
    }
}
 
 
三、攔截器:普通的JavaBean,在呼叫業務方法的之前或者之後會自動攔截並執行自己的一些方法。
 
/**
* Created by IntelliJ IDEA.
* User: leizhimin
* Date: 2008-3-20 23:26:31
* Company: LavaSoft(http://lavasoft.blog.51cto.com/)
* 攔截器
*/

public class InterceptorClass {
    public void before(){
        System.out.println("攔截器InterceptorClass方法呼叫:before()!");
    }
    public void after(){
        System.out.println("攔截器InterceptorClass方法呼叫:after()!");
    }
}
 

四、模擬客戶端:執行業務處理的入口。
 
/**
* Created by IntelliJ IDEA.
* User: leizhimin
* Date: 2008-3-21 0:32:55
* Company: LavaSoft(http://lavasoft.blog.51cto.com/)
* 客戶端
*/

public class Client {
    public static void main(String args[]) {
        DynamicProxyHandler handler = new DynamicProxyHandler();
        BusinessInterface business = new BusinessClass();
        BusinessInterface businessProxy = (BusinessInterface) handler.bind(business);
        businessProxy.doSomething();
    }
}

 

相關文章