注册 | 登录 忘记密码? 51cto首页 | 博客 | 论坛 | 招聘
热点文章 可扩展、高可用、负载均衡..
 帮助

Spring AOP 本质(4)


2008-05-06 16:29:08
 标签:Spring AOP   [推送到技术圈]

版权声明:原创作品,如需转载,请与作者联系。否则将追究法律责任。
Spring AOP 本质(4)
 
这一主要看看Spring AOP是如何实现通知包围的。
 
Spring AOP包围通知在功能上和前置通知加后置通知类似,但还是有区别的:包围通知可以修改返回值,还可以阻止、替换目标方法的执行。
 
Spring里的包围通知是实现MethodInterceptor接口的拦截器。
 
Spring包围通知有着很广泛的应用,比如远程代理和事务管理,都是由拦截器完成。另外,拦截器也是剖析程序运行的好方法。
 
下面利用Spring AOP包围通知实现监控业务方法的执行运行过程耗时情况。
 
/**
* 业务组件
*/

public class WorkerBean {

    public void doSomeWork(int noOfTimes) {
        for(int x = 0; x < noOfTimes; x++) {
            work();
        }
    }
    
    private void work() {
        System.out.print("");
    }
}
 
import java.lang.reflect.Method;

import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;
import org.springframework.util.StopWatch;

/**
* 拦截器,实现方法包围通知
*/

public class ProfilingInterceptor implements MethodInterceptor {

    public Object invoke(MethodInvocation invocation) throws Throwable {
        //启动一个 stop watch
        StopWatch sw = new StopWatch();
        //运行计时器
        sw.start(invocation.getMethod().getName());
        //执行业务方法
        Object returnValue = invocation.proceed();
        //停止计时器
        sw.stop();
        //垃圾信息输出
        dumpInfo(invocation, sw.getTotalTimeMillis());
        //返回业务方法返回值
        return returnValue;
    }

    /**
     * 垃圾信息输入方法,实际上输出的是方法运行的计时信息
     */

    private void dumpInfo(MethodInvocation invocation, long ms) {
        //获取被调用方法
        Method m = invocation.getMethod();
        //获取被调用方法所属的对象
        Object target = invocation.getThis();
        //获取被调用方法的参数
        Object[] args = invocation.getArguments();

        System.out.println("所执行的方法: " + m.getName());
        System.out.println("对象的类型: " + target.getClass().getName());

        System.out.println("方法的参数:");
        for (int x = 0; x < args.length; x++) {
            System.out.print("    > " + args[x]);
        }
        System.out.print("\n");

        System.out.println("抓取方法运行的时间: " + ms + " ms");
    }
}
 
import org.springframework.aop.framework.ProxyFactory;

/**
* 客户端测试方法
*/

public class ProfilingExample {

    public static void main(String[] args) {
        //创建代理对象
        WorkerBean bean = getWorkerBean();
        //在代理对象上调用业务方法
        bean.doSomeWork(10000000);
    }

    /**
     * 代理对象工厂
     */

    private static WorkerBean getWorkerBean() {
        //创建目标对象
        WorkerBean target = new WorkerBean();
        //构建代理对象工厂
        ProxyFactory factory = new ProxyFactory();
        factory.setTarget(target);
        factory.addAdvice(new ProfilingInterceptor());

        //生产一个代理对象
        return (WorkerBean)factory.getProxy();
    }
}
 
运行结果:
- Using JDK 1.4 collections
所执行的方法: doSomeWork
对象的类型: com.apress.prospring.ch6.profiling.WorkerBean
方法的参数:
    > 10000000
抓取方法运行的时间: 3453 ms

Process finished with exit code 0
 
从输出的结果中,可以看到程序方法调用的方法名、参数、所属类,以及执行方法所耗费的时间。
 
另外说一下org.springframework.util.StopWatch类,这是个计时器类,此工具类主要可以获取计时器类start()和stop()两次方法调用间的时间。具体可以查看Spring的API文档。另外,我也在apache commons 包里面也有org.apache.common.lang.time.StopWatch。
 
 
 

本文出自 “熔 岩” 博客,转载请与作者联系!



上一篇 Spring AOP 本质(3)  下一篇 Spring AOP 本质(5)



    文章评论
 
2008-05-07 10:11:00
写的好

 

发表评论

昵   称:
验证码:  点击图片可刷新验证码  博客过2级,无需填写验证码
内   容: