简述AOP

AOP(Aspect-OrientedProgramming,面向方面编程),可以说是OOP(Object-Oriented Programing,面向对象编程)的补充和完善。OOP引入封装、继承和多态性等概念来建立一种对象层次结构,用以模拟公共行为的一个集合。当我们需要为分散的对象引入公共行为的时候,OOP则显得无能为力。也就是说,OOP允许你定义从上到下的关系,但并不适合定义从左到右的关系。例如日志功能。日志代码往往水平地散布在所有对象层次中,而与它所散布到的对象的核心功能毫无关系。对于其他类型的代码,如安全性、异常处理和透明的持续性也是如此。这种散布在各处的无关的代码被称为横切(cross-cutting)代码,在OOP设计中,它导致了大量代码的重复,而不利于各个模块的重用。而AOP技术则恰恰相反,它利用一种称为“横切”的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其名为“Aspect”,即切面。所谓“切面”,简单地说,就是将那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可操作性和可维护性。

使用“横切”技术,AOP把软件系统分为两个部分:核心关注点和横切关注点。业务处理的主要流程是核心关注点,与之关系不大的部分是横切关注点。横切关注点的一个特点是,他们经常发生在核心关注点的多处,而各处都基本相似。比如权限认证、日志、事务处理。Aop 的作用在于分离系统中的各种关注点,将核心关注点和横切关注点分离开来。正如Avanade公司的高级方案构架师Adam Magee所说,AOP的核心思想就是“将应用程序中的商业逻辑同对其提供支持的通用服务进行分离。”

AOP相关概念

        切面(aspect):用来切插业务方法的类。

连接点(joinpoint):是切面类和业务类的连接点,其实就是封装了业务方法的一些基本属性,作为通知的参数来解析。

通知(advice):在切面类中,声明对业务方法做额外处理的方法。

切入点(pointcut):业务类中指定的方法,作为切面切入的点。其实就是指定某个方法作为切面切的地方。

目标对象(target object):被代理对象。

AOP代理(aop proxy):代理对象。

通知:

前置通知(before advice):在切入点之前执行。

后置通知(after returning advice):在切入点执行完成后,执行通知。

环绕通知(around advice):包围切入点,调用方法前后完成自定义行为。

异常通知(after throwing advice):在切入点抛出异常后,执行通知。

Weave(织入):指将aspects应用到target对象并导致proxy对象创建的过程称为织入.

PS:织入是关键,通过织入语法,产生proxy对象,如果类有接口就通过JDK创建代理对象,如果类没有接口则使用cglib创建代理对象。因为电脑JDK的版本,我使用的是Spring4.0。但因为我只学习了Spring3.2,所以我是拿着Spring4.0写着Spring3.2的代码。另外因为我们可能要经常修改服务的需求,例如测一下代码运行时间,所以使用XML的方式来实现AOP比注解的方式更好(毕竟如果我们没有源码,使用注解的方式不能在进行修改服务)。

示例:

配置文件:

<?xml version="1.0"encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
:xsi="http://www.w3.org/2001/XMLSchema-instance"
:context="http://www.springframework.org/schema/context"
:aop="http://www.springframework.org/schema/aop"
:schemaLocation="http://www.springframework.org/schema/beans 
       http://www.springframework.org/schema/beans/spring-beans-4.0.xsd
       http://www.springframework.org/schema/context
       http://www.springframework.org/schema/context/spring-context-4.0.xsd
       http://www.springframework.org/schema/aop 
 >
<context:component-scan base-package="net.zmcheng"/>
<bean id=" logInterceptor"class="net.zmcheng.aop.LogInterceptor"/>
<aop:config>
<aop:pointcut expression="execution(public * net.zmcheng.serviceImpl.UserServiceImpl.add())"id="servicePointCut"/>
<aop:aspect id="logAspect"ref=" logInterceptor">
<aop:before method="before"pointcut-ref="servicePointCut"/>
<aop:around method="aroundMethod"pointcut-ref="servicePointCut"/>
</aop:aspect>
</aop:config>
  
</beans>

 

需要使用的jar包:

切面类:

 

packagenet.zmcheng.aop;
 
importorg.aspectj.lang.ProceedingJoinPoint;
 
publicclassLogInterceptor{
 
publicvoidmyMethod(){};
 
publicvoidbefore(){
System.out.println("method start");
}
 
publicvoidaroundMethod(ProceedingJoinPoint pjp){
System.out.println("around start");
try{
pjp.proceed();
}catch(Throwablee){
e.printStackTrace();
}    
System.out.println("around end");
}
    
}

服务层:

packagenet.zmcheng.service;
 
publicinterfaceUserService{
publicvoidadd();
}
 
packagenet.zmcheng.serviceImpl;
importjavax.annotation.Resource;
importorg.springframework.stereotype.Component;
importnet.zmcheng.dao.UserDao;
importnet.zmcheng.service.UserService;
@Component
publicclassUserServiceImplimplementsUserService{
 
privateUserDao userDao;
publicUserDao getUserDao(){
returnuserDao;
}
@Resource
publicvoidsetUserDao(UserDao userDao){
this.userDao=userDao;
}
publicvoidadd(){
userDao.add();
}
 
}

 

DAO层:

packagenet.zmcheng.dao;
 
publicinterfaceUserDao{
publicvoidadd();
}
packagenet.zmcheng.daoImpl;
 
importorg.springframework.stereotype.Component;
 
importnet.zmcheng.dao.UserDao;
@Component
publicclassUserDaoImplimplementsUserDao{
publicvoidadd(){
System.out.println("添加员工成功");
}
}

 

测试类:

packagenet.zmcheng.demo.test;
 
importorg.junit.Test;
importorg.springframework.context.ApplicationContext;
importorg.springframework.context.support.ClassPathXmlApplicationContext;
 
importnet.zmcheng.service.UserService;
importnet.zmcheng.serviceImpl.UserServiceImpl;
 
publicclassSpringTest{
@Test
publicvoidT(){
ApplicationContext ctx=newClassPathXmlApplicationContext("beans.xml");
UserService us=(UserService)ctx.getBean("userServiceImpl");
us.add();
}
}

 

执行结果:
method start
around start
添加员工成功
around end