1. 需求:统计方法执行的性能情况(来源:《精通Spring 4.x》)

// 性能监视类 PerformanceMonitor
package com.noodles.proxy;
public class PerformanceMonitor{
// 通过一个ThreadLocal, 保存与调用线程相关的性能监视信息
private static ThreadLocal<MethodPerformance> performanceRecord =
new ThreadLocal<MethodPerformance>(); // 开始对某一方法进行性能监视
public static void begin(String method){
System.out.println("性能监视开始...");
MethodPerformance mp = new MethodPerformance(method);
performanceRecord.set(mp);
} // 对某一方法的性能监视结束
public static void end(){
System.out.println("性能监视结束...");
MethodPerformance mp = performanceRecord.get();
// 打印性能监视结果信息
mp.printPerformance();
}
} // 记录性能监视类 MethodPerformance
public class MethodPerformance {
private long begin;
private long end;
private String serviceMethod; // 构造函数
public MethodPerformance(String serviceMethod){
this.serviceMethod = serviceMethod;
// 程序执行的开始时间
this.begin = System.currentTimeMillis();
} // 打印方法执行消耗的时间
public void printPerformance(){
end = System.currentTimeMillis();
long elapse = end - begin;
System.out.println(serviceMethod + " 花费 " + elapse + "毫秒");
}
} // 业务类中完成性能监视的功能
public class ForumServiceImpl implements ForumService {
// 方法一: removeTopic
public void removeTopic(int topicId){
// 开始性能监视
PerformanceMonitor.begin(
"com.noodles.proxy.ForumServiceImpl.removeTopic");
System.out.println("模式删除Topic:" + topicId);
try{
Thread.currentThread().sleep(20);
}catch(Exception ex){
throw new RuntimeException(e);
}
// 结束性能监视
PerformanceMonitor.end();
} // 方法二: 根据id,更新个人信息
public void updateInfoById(int userId){
// 开始性能监视
PerformanceMonitor.begin("com.noodles.proxy.ForumServiceImpl.updateInfoById");
System.out.println("模拟更新个人信息:" + userId);
try{
Thread.currentThead().sleep(40);
}catch(Exception ex){
throw new RuntimeException(ex);
}
// 结束性能监视
PerformanceMonitor.end();
}
}

1.2 上述代码存在问题

  • ForumServiceImpl中每个方法都存在性能监视的代码,存在冗余;
  • 性能监视与ForumServiceImpl业务不相关,属于系统功能,此处,存在耦合;
  • AOP(Aspect Oriented Programming, 面向切面编程)也就是为解决此类问题,采用横向抽取的机制,将大量重复,且与业务不相关的代码(如性能监控,事务管理以及日志记录等)剥离,作为一个切面类;

2. AOP 详解

  • AOP 的核心为切面,切面 = 切点 + 增强;
  • 增强(Advice): 从业务类中剥离处理的非业务代码(如上例中的性能监控代码);
  • 切点(Pointcut): 用于解决在业务类中的哪个位置来执行增强(Advice);

2.1 AOP 底层技术

  • AOP 底层依赖的是动态代理技术;根据所代理的类有无实现接口,分为:

    - JDK 动态代理: Proxy类和InvocationHandler

    - CGLib 动态代理:
/**
*JDK 动态代理
*/
// 实现InvocationHandler接口
public class PerformanceHandler implements InvocationHandler{
private Object target; // target 为业务类
public PerformanceHandler(Object target){
this.target = target;
} // 实现 invoke()方法, 返回代理对象
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
// 开启性能监视
PerformanceMonitor.begin(
target.getClass().getName() + "." + method.getName());
// 通过反射调用业务方法
Object obj = method.invoke(target, args);
// 结束性能监视
PerformanceMonitor.end();
return obj;
}
} // 编写测试类 ForumServiceTest
public class ForumServiceTest {
// 希望被代理的目标业务类
ForumService target = new ForumServiceImpl(); // 将目标业务类和横切代码编织到一起
PerformanceHandler handler = new PerformanceHandler(target); // 创建代理类
ForumService proxy = Proxy.newProxyInstance(
target.getClass().getClassLoader(),
target.getClass().getInterfaces(),
handler); // 调用代理实例
proxy.removeTopic(1011);
proxy.updateInfoById(567);
} /**
* CGLib 代理
* 采用底层的字节码技术,可以为一个类创建子类,在子类中采用方法拦截的技术拦截所有父类方法的调用并顺势织入
* 横切逻辑;
*/
public class CglibProxy implements MethodInterceptor {
private Enhancer enhancer = new Enhancer();
public Object getProxy(Class clazz){
enhancer.setSuperClass(clazz);
enhancer.setCallback(this);
return enhancer.create();
} // 拦截父类所有方法的调用
public Object intercept(Object obj, Method method, Object[] args,
MethodProxy proxy) throws Throwable {
PerformanceMonitor.begin(obj.getClass().getName() + "." + method.getName());
Object result = proxy.invokeSuper(obj, args);
PerformanceMonitor.end();
return result;
}
} // 编写测试方法 ForumServiceTest
public class ForumServiceTest {
@Test
public void proxy(){
CglibProxy proxy = new CglibProxy();
ForumServiceImpl forumService =
(ForumServiceImpl)proxy.getProxy(ForumServiceImpl.class);
forumService.removeTopic(1011);
forumService.updateInfoById(567);
}
}

参考资料:

AOP 详解的更多相关文章

  1. Spring4 AOP详解

    Spring4 AOP详解 第一章Spring 快速入门并没有对Spring4 的 AOP 做太多的描述,是因为AOP切面编程概念不好理解.所以这章主要从三个方面详解AOP:AOP简介(了解),基于注 ...

  2. 【转载】Spring AOP详解 、 JDK动态代理、CGLib动态代理

    Spring AOP详解 . JDK动态代理.CGLib动态代理  原文地址:https://www.cnblogs.com/kukudelaomao/p/5897893.html AOP是Aspec ...

  3. [Spring学习笔记 5 ] Spring AOP 详解1

    知识点回顾:一.IOC容器---DI依赖注入:setter注入(属性注入)/构造子注入/字段注入(注解 )/接口注入 out Spring IOC容器的使用: A.完全使用XML文件来配置容器所要管理 ...

  4. Spring AOP详解(转载)所需要的包

    上一篇文章中,<Spring Aop详解(转载)>里的代码都可以运行,只是包比较多,中间缺少了几个相应的包,根据报错,几经百度搜索,终于补全了所有包. 截图如下: 在主测试类里面,有人怀疑 ...

  5. Spring AOP详解及简单应用

    Spring AOP详解   一.前言 在以前的项目中,很少去关注spring aop的具体实现与理论,只是简单了解了一下什么是aop具体怎么用,看到了一篇博文写得还不错,就转载来学习一下,博文地址: ...

  6. 转:Spring AOP详解

    转:Spring AOP详解 一.前言 在以前的项目中,很少去关注spring aop的具体实现与理论,只是简单了解了一下什么是aop具体怎么用,看到了一篇博文写得还不错,就转载来学习一下,博文地址: ...

  7. Spring全家桶——SpringBoot之AOP详解

    Spring全家桶--SpringBoot之AOP详解 面向方面编程(AOP)通过提供另一种思考程序结构的方式来补充面向对象编程(OOP). OOP中模块化的关键单元是类,而在AOP中,模块化单元是方 ...

  8. Spring Aop 详解二

    这是Spring Aop的第二篇,案例代码很详解,可以查看https://gitee.com/haimama/java-study/tree/master/spring-aop-demo. 阅读前,建 ...

  9. AOP详解之三-创建AOP代理后记,创建AOP代理

    AOP详解之三-创建AOP代理后记,创建AOP代理. 上篇文章已经获取到了AOP的信息,接下来就是拿着这些AOP的信息去创建代理了. 首先我们看下创建AOP代理的入口处. //这个方法将返回代理类 p ...

随机推荐

  1. 【AI】Ubuntu NVIDIA CUDA CUDNN安装配置

    https://blog.csdn.net/qq_33200967/article/details/80689543 https://blog.csdn.net/sinat_29963957/arti ...

  2. Jenkins构建触发器

       我们在执行Jenkins的项目构建的时候一般都是通过web管理界面中的”构建”来执行项目构建操作,但是除此之外我们还可以通过项目配置中的”构建触发器”来触发构建操作, 其中”构建触发器”有一种方 ...

  3. LINUX安装中文输入法和那些大坑

    明明有很多事要做,却偏偏不知道要做什么,这种感觉,很令人上火. 一.基础知识 在原生ubuntu14.04英文环境系统中只有IBus拼音,真的好难用.由于搜狗输入法确实比Linux系统下其它的中文输入 ...

  4. nodeJs学习过程之认识nodejs

    nodeJs只是听说过,没见过,没玩过.从新手开始,我想慢慢掌握它. 一.首先,我想知道nodeJs是什么? 摘自百度百科: Node.js是一套用来编写高性能网络服务器的JavaScript工具包, ...

  5. 【EF框架】EF DBFirst 快速生成数据库实体类 Database1.tt

    现有如下需求,数据库表快速映射到数据库实体类 VS给出的两个选择都有问题,并不能实现,都是坑啊 EF .x DbContext 生成器 EF .x DbContext 生成器 测试结果如下 生成文件 ...

  6. springbatch---->springbatch的使用(二)

    这里我们对springbatch做一个比较深入的学习例子,解压文件,读取文件内容过滤写入到数据库中.如果你掉进了黑暗里,你能做的,不过是静心等待,直到你的双眼适应黑暗. springbatch的使用案 ...

  7. C#串口介绍以及简单串口通信程序设计实现

    C#串口介绍以及简单串口通信程序设计实现 周末,没事干,写个简单的串口通信工具,也算是本周末曾来过,废话不多,直接到主题 串口介绍 串行接口简称串口,也称串行通信接口或串行通讯接口(通常指COM接口) ...

  8. [转]13 Hours: The Secret Soldiers of Benghazi

    转:http://www.imfdb.org/wiki/13_Hours:_The_Secret_Soldiers_of_Benghazi The following weapons were use ...

  9. ubuntu14.04_install_gitlab_platform

    /** author: lihaibo URL:http://www.cnblogs.com/horizonli/p/5321770.html */ 下面是干货 [第一部分 安装] 环境:ubuntu ...

  10. svn版本管理

    代码发布方案: 1,安装,优化 软件环境,(nginx,lvs)  <-------运维工程师 2,程序代码(不断更新).   <--------开发工程师,(开发,运维都可以发布) 3, ...