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

前言:
前面我们简单介绍了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开发中的相关术语

  • 关注点:重复代码就叫做关注点即公共业务。

  • Joinpoint(连接点):所谓连接点是指那些被拦截到的点。在Spring中,这些点指的是方法,因为Spring只支持方法类型的连接点。

  • Pointcut(切入点):所谓切入点是指我们要对哪些Joinpoint(方法)进行拦截的定义。

  • Advice(通知/增强):所谓通知是指拦截到Joinpoint(方法)之后所要做的事情就是通知。通知分为前置通知、后置通知、异常通知、最终通知和环绕通知(切面要完成的功能)。

  • Aspect(切面):是切入点和通知的结合。

  • Target(目标对象):代理的目标对象(要增强的类)

  • Weaving(织入):是指把增强应用到目标对象来创建新的代理对象的过程。Spring采用动态代理织入,而AspectJ采用编译期织入和类装载期织入。

  • Proxy(代理):一个类被AOP织入增强后,就产生一个结果代理类。
    Introduction(引介):引介是一种特殊的通知在不修改类代码的前提下,Introduction可以在运行期为类动态地添加一些方法或Field。
    我们还是用上面的例子解释一下这些术语:

  • 连接点:就是里面的三个方法

  • 切入点:假设里面的add()方法加入了日志的功能,这个就add()方法就是切入点

  • 通知/增强:比如增强UserService类里面的add方法,在add方法中添加了日志功能,这个日志功能就称为增强。
    通知类型:

  • 前置通知:在增强的方法执行之前进行操作。

  • 后置通知:在增强的方法执行之后进行操作。

  • 环绕通知:在增强的方法执行之前和执行之后进行操作。

  • 最终通知:增强了两个方法,执行第一个方法,执行第二个方法,在第二个方法执行之后进行操作。
    也可理解为后置通知后面执行的通知或者无论目标方法是否出现异常,最终通知都会执行。
    异常通知:程序出现异常之后执行的通知。

  • 切面:假设有个日志功能,把日志功能加到add()方法上面的过程

  • 目标对象:UserService

  • 织入:把日志功能增加到目标对象(UserService)的过程

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方法:execution(public .(..))
  • 匹配指定包下所有类的方法:execution(* cn.itcast.dao.*(..)),但不包含子包
  • execution(* cn.itcast.dao..(..)),..表示包、子孙包下所有类。
  • 匹配指定类所有方法:execution(* cn.itcast.service.UserService.*(..))
  • 匹配实现特定接口的所有类的方法:execution(* cn.itcast.dao.GenericDAO+.*(..))
  • 匹配所有save开头的方法:execution(* save*(..))
  • 匹配所有类里面的所有的方法: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("方法执行后" );
    }

}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,470评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,393评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,577评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,176评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,189评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,155评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,041评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,903评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,319评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,539评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,703评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,417评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,013评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,664评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,818评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,711评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,601评论 2 353

推荐阅读更多精彩内容