前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Spring(3)——AOP

Spring(3)——AOP

作者头像
羊羽shine
发布2019-05-29 18:32:29
3830
发布2019-05-29 18:32:29
举报
文章被收录于专栏:Golang开发Golang开发

作用:在程序运行期间,不修改源码对已有方法进行增强。 优势:1减少重复代码,2提高开发效率,3维护方便

JDK动态代理
基本概述

Joinpoint(连接点) 所谓连接点是指那些被拦截到的点。在spring中,这些点指的是方法,因为spring只支持方法类型的连接点。 Pointcut(切入点) 所谓切入点是指我们要对哪些Joinpoint进行拦截的定义。 Advice(通知 /增强 ) 所谓通知是指拦截到 Joinpoint 之后所要做的事情就是通知。 通知的类型:前置通知,后置通知,异常通知,最终通知,环绕通知。 Introduction(引介) 引介是一种特殊的通知在不修改类代码的前提下, Introduction可以在运行期为类动态地添加一些方法或 Field。 Target(目标对象) 代理的目标对象。被代理对象 Weaving(织入) 是指把增强应用到目标对象来创建新的代理对象的过程。spring采用动态代理织入,而 AspectJ 采用编译期织入和类装载期织入。 Proxy(代理) 一个类被AOP织入增强后,就产生一个结果代理类。 Aspect(切面) 是切入点和通知(引介)的结合。 mvn 配置

代码语言:javascript
复制
<dependencies>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>5.0.2.RELEASE</version>
        </dependency>

        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.9.3</version>
        </dependency>

    </dependencies>

xml切面配置 切入点表达式: 访问修饰符 返回值 包名.类名.方法名(参数列表) 其中访问修饰符可以省略 *表示通配符 全通配写法* *..*.*(..)

代码语言:javascript
复制
<?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:aop="http://www.springframework.org/schema/aop"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        http://www.springframework.org/schema/context/spring-context.xsd
        http://www.springframework.org/schema/aop
        http://www.springframework.org/schema/aop/spring-aop.xsd">
    <bean id="accountService" class="com.bx.service.impl.AccountServiceImpl"></bean>
    <bean id="logger" class="com.bx.utils.Logger"></bean>

    <aop:config>
        <!--配置切面 -->
        <aop:aspect id="logAdvice" ref="logger">
            <!-- 配置通知的类型,并且建立通知方法和切入点方法的关联-->
            <aop:before method="printLog" pointcut="execution(* com.bx.service.impl.*.*(..))"></aop:before>
        </aop:aspect>
    </aop:config>
</beans>

设置不同通知类型

代码语言:javascript
复制
 <aop:config>
        <!--配置切面 -->
        <aop:pointcut id="service" expression="execution(* com.bx.service.impl.*.*(..))"/>
        <aop:aspect id="logAdvice" ref="logger">
            <!-- 配置通知的类型,并且建立通知方法和切入点方法的关联-->
            <!--前置通知-->
            <aop:before method="beforePrintLog" pointcut-ref="service"></aop:before>
            <!--后置通知-->
            <aop:after-returning method="returnPrintLog" pointcut-ref="service"></aop:after-returning>
            <!--异常通知-->
            <aop:after-throwing method="throwPrintLog" pointcut-ref="service"></aop:after-throwing>
            <!--最终通知-->
            <aop:after method="afterPrintLog" pointcut-ref="service"></aop:after>
        </aop:aspect>
    </aop:config>

spring的环绕通知提供了一种在代码中手动控制增强方式何时执行的方式。 spring框架为我们提供了接口ProceedingJoinPoint,其接口方法proceed(),此方法相当于明确切入点方法,该接口可以作为环绕通知的方法参数,在程序执行是spring框架会提供该接口的的实现类。 在proceed()之前的函数就是前置通知,在之后的就是后置通知,finally就是最终通知

代码语言:javascript
复制
   public Object aroundLog(ProceedingJoinPoint pjp){
        Object rtValue = null;
        try{
            System.out.println("before  print log");
            Object [] args = pjp.getArgs();
            rtValue = pjp.proceed(args);
            System.out.println("after return print log");
            return rtValue;
        }catch (Throwable t){
            throw  new  RuntimeException(t);
        }finally {
            System.out.println("after print log");
        }
    }
AOP注解

设置bean.xml配置

代码语言:javascript
复制
<?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:aop="http://www.springframework.org/schema/aop"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/aop
        http://www.springframework.org/schema/aop/spring-aop.xsd
        http://www.springframework.org/schema/context
        http://www.springframework.org/schema/context/spring-context.xsd">

    <!-- 配置spring创建容器时要扫描的包-->
    <context:component-scan base-package="com.bx"></context:component-scan>

    <!-- 配置spring开启注解AOP的支持 -->
    <aop:aspectj-autoproxy></aop:aspectj-autoproxy>
</beans>

注解在后置通知调用顺序有问题 ,可以使用环绕通知

代码语言:javascript
复制
@Component("logger")
@Aspect //表示是切面类
public class Logger {
    @Pointcut("execution(* com.bx.service.impl.*.*(..))")
    private void pointLogger(){
        
    }

    @Before("pointLogger()")
    public void beforePrintLog(){
        System.out.println("before print log");
    }

    @After("pointLogger()")
    public void returnPrintLog(){
        System.out.println("after return print log");
    }
    @AfterThrowing("pointLogger()")
    public void throwPrintLog(){
        System.out.println("throw print log");
    }

    @After("pointLogger()")
    public void afterPrintLog(){
        System.out.println("after print log");
    }

//    @Around("pointLogger()")
//    public Object aroundLog(ProceedingJoinPoint pjp){
//        Object rtValue = null;
//        try{
//            System.out.println("before  print log");
//            Object [] args = pjp.getArgs();
//            rtValue = pjp.proceed(args);
//            System.out.println("after return print log");
//            return rtValue;
//        }catch (Throwable t){
//            throw  new  RuntimeException(t);
//        }finally {
//            System.out.println("after print log");
//        }
//    }
}
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2019.05.04 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • JDK动态代理
  • 基本概述
  • AOP注解
相关产品与服务
容器服务
腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档