spring架构师

一起来学习Spring:Spring入门与AOP篇

2018-06-03  本文已影响54人  __y

前言:
前面我们简单介绍了Spring的IOC容器,其实,本篇要讲述的是Spring中另外一个常用的模块AOP。在看本篇之前,要确保已经有代理模式的基本知识,这样看起来才不会费劲!

1.AOP简介

AOP(面向切面编程)是对OOP中遇到的一些问题的补充,是OOP的扩展,在不改变原有代码的情况下增加新的功能,比如我们常见的日志功能,权限,异常处理,缓存等公共业务(即与软件的开发业务没有关联)。
举一个简单的例子:

public class UserService{
    //添加用户的方法
      public void add() {
       //加一个输出日志
        log();
          //代码逻辑
      }
       public void delete() {
       //加一个输出日志
        log();
   
          //代码逻辑
      }

}

我们可以看到如果按照我们以前的编写代码的方式的话,每个方法都有写重复的代码,而且log()这个方法还不是我们主要的业务逻辑,这样编码的话充满耦合性,这时候我们就要利用Spring的AOP技术将log()这一块抽取出来
Spring的AOP底层实现
Spring的AOP底层用到了我们两种动态代理模式JDK的动态代理(针对实现了接口的类产生代理),Cglib的动态代理(针对没有实现接口的类产生代理,应用的是底层的字节码增强的技术,生成当前的子类对象)

2.AOP开发中的相关术语

3.使用SpringAOP开发步骤(基于AspectJXML方式)

注意对切面的配置有两种方式:一种是传统的aop联盟制定的aop通知的配置,一种是spring支持的AspectJ框架的切面


image.png

1.引入相关的jar文件

spring-aop-3.2.5.RELEASE.jar 【spring3.2源码】
aopalliance.jar 【spring2.5源码/lib/aopalliance】
aspectjweaver.jar 【spring2.5源码/lib/aspectj】或【aspectj-1.8.2\lib】
aspectjrt.jar 【spring2.5源码/lib/aspectj】或【aspectj-1.8.2\lib】

2.bean.xml中引入aop名称空间

<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"
    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">

3.创建业务功能代码和公共代码

public class TestAop {
    public void sayHello(String name) {
        System.out.println("sayHello方法被执行");
    }
}

public class Log {
    //方法执行前
    public void  before() {
        System.out.println("方法执行前" );
    }
    //方法执行后
    public void  after() {
        System.out.println("方法执行后" );
    }
  
}

4.配置bean.xml

    <!--配置业务Bean-->
        <bean id="testAop" class="test.TestAop"></bean>
        <!--配置切面Bean-->
        <bean id="log" class="test.Log"></bean>
        <!--配置切面-->
        <aop:config>
            <!--定义切入表达式,要拦截什么方法-->
            <aop:pointcut id="pointCut" expression="execution(* test.TestAop.*(..))"></aop:pointcut>
            <!--指定切面是那个类-->
            <aop:aspect ref="log">
                <!--指定来拦截的时候执行切面类的哪些方法-->
                <!-- 配置前置通知 -->
                <aop:before method="before" pointcut-ref="pointCut" ></aop:before>
                <!-- 配置后通知 -->
                <aop:after method="after" pointcut-ref="pointCut" />

            </aop:aspect>
        </aop:config>

结果


image.png

切入点表达式

通过execution函数,可以定义切点的方法切入。
语法为:execution(<访问修饰符>?<返回类型><方法名>(<参数>)<异常>)。
例如:

        <!-- 【拦截所有public方法】 -->
        <!--<aop:pointcut expression="execution(public * *(..))" id="pt"/>-->

        <!-- 【拦截所有save开头的方法 】 -->
        <!--<aop:pointcut expression="execution(* save*(..))" id="pt"/>-->

        <!-- 【拦截指定类的指定方法, 拦截时候一定要定位到方法】 -->
        <!--<aop:pointcut expression="execution(public * cn.itcast.g_pointcut.OrderDao.save(..))" id="pt"/>-->

        <!-- 【拦截指定类的所有方法】 -->
        <!--<aop:pointcut expression="execution(* cn.itcast.g_pointcut.UserDao.*(..))" id="pt"/>-->

        <!-- 【拦截指定包,以及其自包下所有类的所有方法】 -->
        <!--<aop:pointcut expression="execution(* cn..*.*(..))" id="pt"/>-->

        <!-- 【多个表达式】 -->
        <!--<aop:pointcut expression="execution(* cn.itcast.g_pointcut.UserDao.save()) || execution(* cn.itcast.g_pointcut.OrderDao.save())" id="pt"/>-->
        <!--<aop:pointcut expression="execution(* cn.itcast.g_pointcut.UserDao.save()) or execution(* cn.itcast.g_pointcut.OrderDao.save())" id="pt"/>-->
        <!-- 下面2个且关系的,没有意义 -->
        <!--<aop:pointcut expression="execution(* cn.itcast.g_pointcut.UserDao.save()) &amp;&amp; execution(* cn.itcast.g_pointcut.OrderDao.save())" id="pt"/>-->
        <!--<aop:pointcut expression="execution(* cn.itcast.g_pointcut.UserDao.save()) and execution(* cn.itcast.g_pointcut.OrderDao.save())" id="pt"/>-->

        <!-- 【取非值】 -->
        <!--<aop:pointcut expression="!execution(* cn.itcast.g_pointcut.OrderDao.save())" id="pt"/>-->

4.使用注解的形式

在实际开发中,如果我们可以用注解形式的话,还是推荐使用注解的形式

1.在bean.xml中开启AOP注解方式

 <!-- 开启aop注解方式 -->
 <aop:aspectj-autoproxy></aop:aspectj-autoproxy>

切面类

@Aspect//指定为切面类
public class Log {
    //方法执行前
    @Before("execution(* test.TestAop.*(..))")
    public void  before() {
        System.out.println("方法执行前");
    }
    //方法执行后
    @After("execution(* test.TestAop.*(..))")
    public void  after() {
        System.out.println("方法执行后" );
    }

}

image.png

优化:

@Aspect//指定为切面类
public class Log {
    // 指定切入点表达式,拦截哪个类的哪些方法
    // 指定切入点表达式,拦截哪个类的哪些方法
    @Pointcut("execution(* test.TestAop.*(..))")
    public void pt() {

    }
    //方法执行前
    @Before ("pt()")
    public void  before() {
        System.out.println("方法执行前");
    }
    //方法执行后
    @After( "pt()")
    public void  after() {
        System.out.println("方法执行后" );
    }

}
上一篇 下一篇

猜你喜欢

热点阅读