Spring boot之AOP面向切面编程

来源:互联网 发布:listview性能优化 编辑:程序博客网 时间:2024/05/16 16:21

        如果说依赖注入(DI,Dependency Injection)有助于应用对象之间的解耦,那么面向切面编程(AOP,Aspect Oriented Programing)则有助于横切关注点与它们所影响的对象之间的解耦!!!

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

       相关概念:

        连接点(Joinpoint): 表示需要在程序中插入横切关注点的扩展点,连接点可能是类初始化、方法执行、方法调用、字段调用或处理异常等等,Spring只支持方法执行连接点;在AOP中表示为“在哪里干”

        切入点(Pointcut): 选择一组相关连接点的模式,即可以认为连接点的集合,Spring支持perl5正则表达式和AspectJ切入点模式,Spring默认使用AspectJ语法;在AOP中表示为“在哪里干的集合”

        通知(Advice):在连接点上执行的行为,通知提供了在AOP中需要在切入点所选择的连接点处进行扩展现有行为的手段;包括前置通知(before advice)、后置通知(after advice)、环绕通知(around advice),在Spring中通过代理模式实现AOP,并通过拦截器模式以环绕连接点的拦截器链织入通知;在AOP中表示为“干什么”

        切面(Aspect):横切关注点的模块化,比如日志组件。可以认为是通知、引入和切入点的组合;在Spring中可以使用Schema和@AspectJ方式进行组织实现;在AOP中表示为“在哪干和干什么集合”
        引入(Introduction): 也称为内部类型声明,为已有的类添加额外新的字段或方法,Spring允许引入新的接口(必须对应一个实现)到所有被代理对象(目标对象);在AOP中表示为“干什么(引入什么)”;
        目标对象(Target Object):需要被织入横切关注点的对象,即该对象是切入点选择的对象,需要被通知的对象,从而也可称为“被通知对象”;由于Spring AOP 通过代理模式实现,从而这个对象永远是被代理对象;在AOP中表示为“对谁干”
        AOP代理(AOP Proxy): AOP框架使用代理模式创建的对象,从而实现在连接点处插入通知(即应用切面),就是通过代理来对目标对象应用切面。在Spring中,AOP代理可以用JDK动态代理或CGLIB代理实现,而通过拦截器模型应用切面。
        织入(Weaving): 织入是一个过程,是将切面应用到目标对象从而创建出AOP代理对象的过程,织入可以在编译期、类装载期、运行期进行。组装方面来创建一个被通知对象。这可以在编译时完成(例如使用AspectJ编译器),也可以在运行时完成。Spring和其他纯Java AOP框架一样,在运行时完成织入。

这里以获取HTTP请求中的字段为例进行解读:

1、在pom.xml中引用spring-boot-starter-aop依赖


<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-aop</artifactId></dependency>
2、创建HttpAspect.java类,加上@Aspect注解

@Poincut注解表示切入的点,即程序中通用的逻辑业务,这里是请求的路径

@Before注解表示当前方法是在具体的请求方法之前执行

@After注解表示当前方法是在具体的请求方法之后 执行

@AfterReturning注解可以得到请求的参数

package com.example.aspect;import org.aspectj.lang.annotation.*;import org.aspectj.lang.JoinPoint;import org.slf4j.Logger;import org.slf4j.LoggerFactory;import org.springframework.stereotype.Component;import org.springframework.web.context.request.RequestContextHolder;import org.springframework.web.context.request.ServletRequestAttributes;import javax.servlet.http.HttpServletRequest;/** * 利用AOP技术对往数据库中插入学生信息作权限验证 */@Aspect@Componentpublic class HttpAspect {    private final static Logger logger = LoggerFactory.getLogger(HttpAspect.class);    //@Before的注解在方法执行之前执行    //拦截该路径下studentList()方法,两个点表示任何参数//    @Before("execution(public * com.example.controller.StudentController.studentList(..))")//    public void log(){//        System.out.println("Test before");//    }////    @After("execution(public * com.example.controller.StudentController.studentList(..))")//    public void doAfter(){//        System.out.println("Test after");//    }    //定义一个公用方法    @Pointcut("execution(public * com.example.controller.StudentController.*(..))")    public void log(){    }    @Before("log()")    public void doBefore(JoinPoint joinPoint){        ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();        HttpServletRequest request = attributes.getRequest();        //url        logger.info("url={}", request.getRequestURI());        //method        logger.info("method={}", request.getMethod());        //ip        logger.info("ip={}", request.getRemoteAddr());        //method        logger.info("class_method={}", joinPoint.getSignature().getDeclaringTypeName() + "." + joinPoint.getSignature().getName());        //param        logger.info("args={}", joinPoint.getArgs());    }    @After("log()")    public void doAfter(){    }    @AfterReturning(returning="obj", pointcut = "log()")    public void doAfterReturnig(Object obj){        logger.info("reponse={}", obj);    }}


Git代码

参考资料:

1、http://www.imooc.com/learn/767

2、http://jinnianshilongnian.iteye.com/blog/1482071下AOP相关内容;



        

原创粉丝点击