AOP及spring AOP的使用

介绍

AOP是一种概念(思想),并没有设定具体语言的实现。
AOP是对oop的一种补充,不是取而代之。
具体思想:定义一个切面,在切面的纵向定义处理方法,处理完成之后,回到横向业务流。

特征

散布于应用中多处的功能被称为横切关注点
横切关注点可以被描述为影响应用多处的功能
切面提供了取代继承和委托的另一种可选方案
通过声明的方式定义这个功能要以何种方式在何处应用,而无需修改受影响的类

处理过程

核心概念

连接点(join point)

就是程序执行的某个特定的点:

你的一个类里,有15个方法,那就有几十个连接点了
Spring仅支持方法的连接点
连接点就是为了让你好理解切点,搞出来的,明白这个概念就行了

切点(pointcut)

顾名思义切点就是切入哪些连接点,
切点是用来筛选连接点的,来定义都有哪些连接点需要被执行织入通知,
连接点相当于数据库中的记录,而切点相当于查询条件,
一个切点可以匹配出多个连接点
切点指明:都有哪些切点被切面选中了

通知(advice)

切面的工作内容被称为通知,主要是定义我们要做什么事情
比如:我们要记录每一个方法执行前后的时间日志,这个就是我们这个aop的切面工作内容,也就是通知
基于连接点的概念,我们必须要指定这些切点
一般spring的切点都是匹配到方法级别,不会直接定义到某个类某个方法的哪个点,
所以我们需要为这些切点指定在什么时候织入通知。
比如:我们的切点包括了10类的100个方法,那么我们不会为每一个类中的方法单独指定在哪个点(前,后,异常)执行通知,我们会采用统一规则,比如都指定为方法前。
方法前(before)
方法后(after)
方法前后(around)
跑出异常(Exception)
通知指明:做什么事情,在什么时机(统一切入点)执行

切面

通知和切点的结合
通知和切点共同定义了切面的全部内容——它是什么,在何时和何处完成其功能

引入

允许我们向现有的类添加新方法属性。这不就是把切面(也就是新方法属性:通知定义的)用到目标类中吗

织入

织入是把切面应用到目标对象并创建新的代理对象的过程
编译期:切面在目标类编译时被织入。这种方式需要特殊的编译器。AspectJ的织入编译器就是以这种方式织入切面的。
类加载期:切面在目标类加载到JVM时被织入。这种方式需要特殊的类加载器(ClassLoader),它可以在目标类被引入应用之前增
强该目标类的字节码。AspectJ 5的加载时织入(load-time weaving,LTW)就支持以这种方式织入切面。
运行期:切面在应用运行的某个时刻被织入。一般情况下,在织入切面时,AOP容器会为目标对象动态地创建一个代理对象。Spring
AOP就是以这种方式织入切面的。

spring mvc中的Aop的实现方式

基于XML配置的Spring AOP

创建一个普通的pojo类,没有添加任何注解也没有继承和实现任何接口

使用xml的方式创建这个切面处理类的实例

声明切点和切面

注意通知的配置

基于注解配置Spring AOP

首先需要在spring的配置文件中启动对@AspectJ注解的支持

创建一个切面处理类为这个类上加上@Aspect以及@Component注解

创建一个切入点表达式

创建前置通知


@Before
@AfterReturning
@AfterThrowing
@After
@Around

实战

项目中有很多的页面都需要展示用户的信息,
没有使用aop之前每一个controller层都需要去设置用户信息,
于是想起来使用aop的方式解决;

创建一个注解,用于标识哪些conroller处理函数需要添加用户信息

/**
 * 注入用户信息
 * <p>
 * 在controller层中添加这个注解就可以实现在返回页面的
 * 的数据里添加上用户的信息
 * 
 * @author   WangSen(wangsenhehe@126.com)
 * @Date     2018年1月17日      
 */
@Target({ ElementType.METHOD, ElementType.TYPE })
@Retention(RetentionPolicy.RUNTIME)
public @interface InjectionUserInfo {
}

编写切面处理类

/**
* PageService.java
* com.we.core.web.aop
* Copyright (c) 
*/

package cn.isuyang.web.www.aop;

import java.lang.reflect.Method;

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;

import cn.isuyang.service.UserDetailBaseService;
import cn.isuyang.web.www.annotation.InjectionUserInfo;

import com.we.bussiness.sso.interfaces.IFetchUser;
import com.we.core.web.util.AspectUtil;
import com.we.core.web.util.MvcUtil;

/**
* 个人中心用户信息切面
*
* @author  wangsen(wangsenhehe@126.com)
* @Date    2018年1月17日
*/
@Component
@Aspect
@Order(2)
public class UserInfoAspect {
    @Autowired
    private UserDetailBaseService userDetailBaseService;
    @Autowired
    private IFetchUser fetchUser;

    /**
    * 
    * @param pjp 执行的切点对象
    * 
    * @throws Throwable 方法执行错误的时候会抛此异常
    */
    @Around("@annotation(cn.isuyang.web.www.annotation.InjectionUserInfo)")
    @Order(1)
    public Object setUserInfos(final ProceedingJoinPoint pjp) throws Throwable {
        Object result = pjp.proceed();
        if (!isNeedSetUserInfo(pjp)) {
            return result;
        }
        MvcUtil.getRequest().setAttribute("userInfo",
                userDetailBaseService.getByUserId(fetchUser.getCurrentUserIdWithEx()));
        return result;
    }

    /**
    * 是否需要设置用户信息
    *
    * @param pjp 切点
    * 
    * @return true表示存在
    */
    private boolean isNeedSetUserInfo(final ProceedingJoinPoint pjp) {
        Method method = AspectUtil.getMethod(pjp);
        return method.isAnnotationPresent(InjectionUserInfo.class);
    }
}

在controller的处理函数上添加 @InjectionUserInfo注解即可

/**
 * 
 * @author   WangSen(wangsenhehe@126.com)
 * @Date     2017年8月10日      
 */
@Controller
@RequestMapping("/personcenter")
public class PersonCenterController {
    @Autowired
    private PersonCenterViewService personCenterViewService;
    /**
     * 我的项目
     *
     * @param model 模型
    */
    @RequestMapping
    @InjectionUserInfo
    public void myProject(final MyProjectListForm listform, @ModelAttribute final Page page, final Model model) {
        personCenterViewService.setMyProjectInfo(listform, page, model);
    }
    /**
     * 我的关注
     *
     * @param model 模型
     */
    @RequestMapping
    @InjectionUserInfo
    public void myFollow(final MyFollowListForm listform, @ModelAttribute final Page page, final Model model) {
        personCenterViewService.setMyFollowInfo(listform, page, model);
    }
}

资料

aop的基本概念
http://blog.csdn.net/yuanye348623610/article/details/8823429
http://www.jianshu.com/p/601713b20e43
了解知识【AspectJ表达式函数或者叫做切入点表达式】
execution():满足匹配模式字符串的所有目标类方法的连接点
@annotation():任何标注了指定注解的目标方法链接点
资料http://blog.csdn.net/wangpeng047/article/details/8556800
spring aop的实现原理
http://www.cnblogs.com/lcngu/p/5339555.html
http://blog.csdn.net/luanlouis/article/details/51155821
spring aop的简单模拟实现
http://www.cnblogs.com/hqbhonker/p/3822237.html

posted @ 2018-01-17 19:10  王森  阅读(942)  评论(0编辑  收藏  举报