本文编写于 124 天前,最后修改于 124 天前,其中某些信息可能已经过时。

普通的编程是自向而下式编程,如果在你中间部分加上其他奇怪的需求就会很麻烦,而AOP的思想就很完美的解决了这个问题。

例如我们想在某个功能上加日志,我们需要找到这个功能作为切入关注点(面),将功能横向加入即可。就像我们把需求写在代理类那样。

(这东西真麻烦,还是用注解舒服)

通过SpringAPI接口实现

我们先来创建一个接口包含CURD

public interface UserService {
    void insert();
    void delete();
    void update();
    void select();
}

然后开始实现这个接口

public class UserServiceImpl implements UserService{
    @Override
    public void insert() {
        System.out.println("插入了xxx");
    }

    @Override
    public void delete() {
        System.out.println("删除了xxx");
    }

    @Override
    public void update() {
        System.out.println("修改了xxx");
    }

    @Override
    public void select() {
        System.out.println("查询了xxx");
    }
}

由于我们使用的是动态代理的思想我们只需要关注接口做什么。

public class Test {
    public static void main(String[] args) {
        ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
        UserService userService = (UserService) context.getBean("userService");
        userService.select();
    }
}
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        https://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop.xsd">

    <context:component-scan base-package="com.springagain01.pojo"/>
    <context:annotation-config/>
    <bean id="userService" class="com.springagain01.service.UserServiceImpl"/>
    <bean id="afterLog" class="com.springagain01.log.AfterLog"/>
    <bean id="beforeLog" class="com.springagain01.log.BeforeLog"/>

    <aop:config>
        <aop:pointcut id="pointcut" expression="execution(* com.springagain01.service.UserServiceImpl.*(..))"/>
        <aop:advisor advice-ref="beforeLog" pointcut-ref="pointcut"/>
        <aop:advisor advice-ref="afterLog" pointcut-ref="pointcut"/>
    </aop:config>
</beans>

那个@m的标示点击。

从这里我们已经可以看出它已经执行了环绕增加

这里讲个坑,因为我直接使用的直接创建SpringBoot项目并且导入依赖,没想到缺了个包,改了一上午bug……aspectjweaver这个包一定一定要导入

给爷爬

整个DIY的新活,上一个导入类这次直接导入方法(切面定义)

public class DiyPointCut {

    public void Before(){
        System.out.println("==============方法执行前==============");
    }
    public void After(){
        System.out.println("==============方法执行后==============");
    }
}
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        https://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop.xsd">

<!--    <context:component-scan base-package="com.springagain01.pojo"/>-->
<!--    <context:annotation-config/>-->
    <bean id="userService" class="com.springagain01.service.UserServiceImpl"/>
    <bean id="diy" class="com.springagain01.diy.DiypointCut"/>

    <aop:config>
        <aop:aspect ref="diy">
            <aop:pointcut id="point" expression="execution(* com.springagain01.service.UserServiceImpl.*(..))"/>
            <aop:before method="Before" pointcut-ref="point"/>
            <aop:after method="After" pointcut-ref="point"/>
        </aop:aspect>
    </aop:config>
</beans>

接下来是喜闻乐见地注解环节

首先先开启一个注解支持,在此之前别忘了把类注入

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        https://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop.xsd">

    <bean id="userService" class="com.springagain01.service.UserServiceImpl"/>
    <bean id="diy" class="com.springagain01.diy.DiypointCut"/>
    <aop:aspectj-autoproxy/>

</beans>

其他无需更改。当然你如果不是很想用bean标签可以把这个类,使用@Component注解,别忘了扫描包和添加注解支持

@Aspect
public class DiypointCut {
    //括号里是切入点
    @Before("execution(* com.springagain01.service.UserServiceImpl.*(..))")
    public void Before(){
        System.out.println("==============方法执行前==============");
    }
    @After("execution(* com.springagain01.service.UserServiceImpl.*(..))")
    public void After(){
        System.out.println("==============方法执行后==============");
    }
    //在环绕增强我们可以给定一个参数,代表我们要获取的切入的点。JoinPoint连接点:与切入点匹配的执行点
    @Around("execution(* com.springagain01.service.UserServiceImpl.*(..))")
    public void Around(ProceedingJoinPoint jp) throws Throwable {
        System.out.println("环绕前");
        Object proceed = jp.proceed();
        System.out.println("环绕后");
    }
}