面向切面编程(也叫面向方面),可以通过预编译方式和运行期动态代理实现在不修改源代码的情况下给程序动态统一添加功能的一种技术。AOP实际是GoF设计模式的延续,设计模式孜孜不倦追求的是调用者和被调用者之间的解耦,AOP可以说也是这种目标的一种实现。
面向切面编程:就是在原直线编程的某处咔嚓一下,加点动西,而且不影响原来的代码。可以用在日志,加权限,事物,异常管理等方面。
从简单的例子说起:
现在有一个save()方法,需要在save()前、后分别加上日志打印的信息。
- public class UserDAOImp implements UserDAO
- {
- @Override
- public void save(User u)
- {
- System.out.println("a DAO implements!");
- }
- }
直接在源代码save里加一些输出就行了。
前提条件:有源代码!
- public class UserDAOImp implements UserDAO
- {
- @Override
- public void save(User u)
- {
- System.out.println("before!!!!!");
- System.out.println("a DAO implements!");
- System.out.println("after!!!!!!");
- }
- }
评价:傻瓜式的!可拓展性差!!
2. 解决办法二: 如果没有源代码呢?
可以继承UserDAOImp, 然后在xml中将bean的class改成继承的类。
前提条件:继承
- public class UserDAOImp1 extends UserDAOImp
- {
- @Override
- public void save(User u)
- {
- System.out.println("save start......");
- super.save(u);
- }
- }
3. 解决办法三:
针对第二种办法优化,用接口。
- public class UserDAOImp2 implements UserDAO
- {
- private UserDAO userDAO = new UserDAOImp();
- @Override
- public void save(User u)
- {
- System.out.println("save start.....");
- userDAO.save(u);
- }
- }
4. 解决办法四:
AOP
在不修改源代码的情况下给程序动态统一添加功能。
- /*
- *面向切面编程, 动态代理. Aspect声明切面, Component初始化.
- */
- @Aspect
- @Component
- public class LogInterceptor
- {
- //这个可用来替代以后重复出现的. 直接在后面的Before("myMethod()")就行了.
- @Pointcut("execution(public * com.dao.impl..*.*(..))")
- public void myMethod(){};
- //下面用到的是织入点语法, 看文档里面有. 就是指定在该方法前执行
- //@Before("execution(public void com.dao.impl.UserDAOImp.save(com.model.User))")
- //记住下面这种通用的, *表示所有
- @Before("execution(public * com.dao.impl..*.*(..))")
- public void beforeMethod()
- {
- System.out.println("save start......");
- }
- //正常执行完后
- @AfterReturning("execution(public * com.dao.impl..*.*(..))")
- public void afterReturnning()
- {
- System.out.println("after save......");
- }
- //抛出异常时才调用
- @AfterThrowing("myMethod()")
- public void afterThrowing()
- {
- System.out.println("after throwing......");
- }
- //环绕, 这个特殊点.
- @Around("myMethod()")
- public void aroundMethod(ProceedingJoinPoint pjp) throws Throwable
- {
- //加逻辑的时候, 不要依赖执行的的先后顺序
- System.out.println("method around start!");
- pjp.proceed();
- System.out.println("method around end!");
- }
- }