面向切面编程(也叫面向方面),可以通过预编译方式和运行期动态代理实现在不修改源代码的情况下给程序动态统一添加功能的一种技术。AOP实际是GoF设计模式的延续,设计模式孜孜不倦追求的是调用者和被调用者之间的解耦,AOP可以说也是这种目标的一种实现。
面向切面编程:就是在原直线编程的某处咔嚓一下,加点动西,而且不影响原来的代码。可以用在日志,加权限,事物,异常管理等方面。
 
从简单的例子说起
现在有一个save()方法,需要在save()前、后分别加上日志打印的信息。
  1. public class UserDAOImp implements UserDAO
  2. {
  3. @Override
  4. public void save(User u)
  5. {
  6. System.out.println("a DAO implements!");
  7. }
  8. }

1.  解决办法一:

直接在源代码save里加一些输出就行了。
前提条件:有源代码!
  1. public class UserDAOImp implements UserDAO
  2. {
  3. @Override
  4. public void save(User u)
  5. {
  6. System.out.println("before!!!!!");
  7. System.out.println("a DAO implements!");
  8. System.out.println("after!!!!!!");
  9. }
  10. }
评价:傻瓜式的!可拓展性差!!

2.  解决办法二:

如果没有源代码呢?
可以继承UserDAOImp, 然后在xml中将bean的class改成继承的类。
前提条件:继承
  1. public class UserDAOImp1 extends UserDAOImp
  2. {
  3. @Override
  4. public void save(User u)
  5. {
  6. System.out.println("save start......");
  7. super.save(u);
  8. }
  9. }

评价:继承尽量少用。可拓展性也差!!

 
3.  解决办法三:
针对第二种办法优化,用接口。
  1. public class UserDAOImp2 implements UserDAO
  2. {
  3. private UserDAO userDAO = new UserDAOImp();
  4. @Override
  5. public void save(User u)
  6. {
  7. System.out.println("save start.....");
  8. userDAO.save(u);
  9. }
  10. }

评价:比较灵活了,但还是有一个问题, 当有1000个bean时, 就要组合1000次........如何解决?

4. 解决办法四:
AOP
在不修改源代码的情况下给程序动态统一添加功能。
  1. /*
  2. *面向切面编程, 动态代理. Aspect声明切面, Component初始化.
  3. */
  4. @Aspect
  5. @Component
  6. public class LogInterceptor
  7. {
  8. //这个可用来替代以后重复出现的. 直接在后面的Before("myMethod()")就行了.
  9. @Pointcut("execution(public * com.dao.impl..*.*(..))")
  10. public void myMethod(){};
  11. //下面用到的是织入点语法, 看文档里面有. 就是指定在该方法前执行
  12. //@Before("execution(public void com.dao.impl.UserDAOImp.save(com.model.User))")
  13. //记住下面这种通用的, *表示所有
  14. @Before("execution(public * com.dao.impl..*.*(..))")
  15. public void beforeMethod()
  16. {
  17. System.out.println("save start......");
  18. }
  19. //正常执行完后
  20. @AfterReturning("execution(public * com.dao.impl..*.*(..))")
  21. public void afterReturnning()
  22. {
  23. System.out.println("after save......");
  24. }
  25. //抛出异常时才调用
  26. @AfterThrowing("myMethod()")
  27. public void afterThrowing()
  28. {
  29. System.out.println("after throwing......");
  30. }
  31. //环绕, 这个特殊点.
  32. @Around("myMethod()")
  33. public void aroundMethod(ProceedingJoinPoint pjp) throws Throwable
  34. {
  35. //加逻辑的时候, 不要依赖执行的的先后顺序
  36. System.out.println("method around start!");
  37. pjp.proceed();
  38. System.out.println("method around end!");
  39. }
  40. }

aop 例子(annotation方式实现)的更多相关文章

  1. 5.3 Spring5源码--Spring AOP使用接口方式实现

    Spring 提供了很多的实现AOP的方式:Spring 接口方式,schema配置方式和注解. 本文重点介绍Spring使用接口方式实现AOP. 使用接口方式实现AOP以了解为目的. 更好地理解动态 ...

  2. spring aop 使用注解方式总结

    spring aop的注解方式:和xml的配置方式略有区别,详细如下: 1.首先还是建立需要的切面类:切面类里面定义好切点配置,以及所有的需要实现的通知方法. /** * */ package com ...

  3. springboot aop 自定义注解方式实现完善日志记录(完整源码)

    版权声明:本文为博主原创文章,欢迎转载,转载请注明作者.原文超链接 一:功能简介 本文主要记录如何使用aop切面的方式来实现日志记录功能. 主要记录的信息有: 操作人,方法名,参数,运行时间,操作类型 ...

  4. springboot aop 自定义注解方式实现一套完善的日志记录(完整源码)

    https://www.cnblogs.com/wenjunwei/p/9639909.html https://blog.csdn.net/tyrant_800/article/details/78 ...

  5. Spring入门6事务管理2 基于Annotation方式的声明式事务管理机制

    Spring入门6事务管理2 基于Annotation方式的声明式事务管理机制 201311.27 代码下载 链接: http://pan.baidu.com/s/1kYc6c 密码: 233t 前言 ...

  6. Spring整合Hibernate:2、使用Annotation方式进行声明式的事务管理

    1.加入DataSourceTransactionManager的命名空间 修改applicationContext.xml文件,增加如下内容: 1 2 3 4 5 6 7 8 9 10 11 12 ...

  7. 一个简单的Spring AOP例子

    转载自: http://www.blogjava.net/javadragon/archive/2006/12/03/85115.html 经过这段日子的学习和使用Spring,慢慢地体会到Sprin ...

  8. spring+springMVC集成(annotation方式)

    spring+springMVC集成(annotation方式) SpringMVC+Spring4.0+Hibernate 简单的整合 MyBatis3整合Spring3.SpringMVC3

  9. 框架源码系列七:Spring源码学习之BeanDefinition源码学习(BeanDefinition、Annotation 方式配置的BeanDefinition的解析)

    一.BeanDefinition 1. bean定义都定义了什么? 2.BeanDefinition的继承体系  父类: AttributeAccessor: 可以在xml的bean定义里面加上DTD ...

随机推荐

  1. java:注解(二)

    自定义注解 使用@interface自定义注解时,自动继承了java.lang.annotation.Annotation接口,由编译程序自动完成其他细节.在定义注解时,不能继承其他的注解或接口.@i ...

  2. 记录MySQL运行的SQL

    对照Oracle功能去学习Mysql总会发现亮点 Oracle中通过日志挖掘这一技能,能够找到以前运行过的全部记录: Mysql中也提供了3种方法{验证过的,我会记录详细做法} 方法1:{已验证} 记 ...

  3. 笔记本WIFI卡简介

    1.Intel AC9560(CNVI) AC9260(pcie) 3165D2W(pcie) 2.Realtek瑞昱 RTL8822be(pcie) RTL8723BU(USB) 英特尔在300系主 ...

  4. Hive报错:Failed with exception Unable to rename

    之前也安装过hive,操作过无数,也没发现什么错误,今天因为之前安装的hadoop不能用了,不知道为什么,老是提示node 0,所以重新安装了hadoop和hive.安装完测试hive创建表也没发现什 ...

  5. 【Python基础】之不同的文件在不同目录下导入指定模块的方法

    如下图三个文件的目录路径 – project     |–  1     |    |–  2     |    |    |–  3    |    |    |    |– owen.py     ...

  6. /usr/bin/mysqld_safe_helper: Cannot change uid/gid (errno: 1) (转)

    From: https://www.rootusers.com/how-to-fix-mariadb-10-0-29-selinux-update-failure/ 安装mysql 10.0.29后, ...

  7. 图像检测之sift and surf---sift中的DOG图 surf hessian

    http://www.cnblogs.com/tornadomeet/archive/2012/08/17/2644903.html http://www.cnblogs.com/slysky/arc ...

  8. 【BZOJ1336】[Balkan2002]Alien最小圆覆盖 随机增量法

    [BZOJ1336][Balkan2002]Alien最小圆覆盖 Description 给出N个点,让你画一个最小的包含所有点的圆. Input 先给出点的个数N,2<=N<=10000 ...

  9. Grunt学习笔记【5】---- expand使用方法

    本文主要讲expand使用方法. 当你希望处理大量的单个文件时,这里有一些附加的属性可以用来动态的构建一个文件列表.这些属性都可以用于 Compact 和 Files Array 文件映射格式. ex ...

  10. Linux就该这么学--Shell脚本基本应用

    1.接收用户的参数: Shell脚本为了能够让用户更灵活的完成工作需求,可以在执行命令时传递参数:(命令名 参数1 参数2...) Shell预定义变量: $0 当前执行Shell脚本的程序名 $1- ...