第二节 - Spring框架 AOP的丧心病狂解说

继续上一节的内容,多几个jar包:

image.png

aop技术是面向切面编程思想,作为OOP的延续思想添加到企业开发中,用于弥补OOP开发过程中的缺陷而提出的编程思想。AOP底层也是面向对象;只不过面向的不是普通的Object对象,而是特殊的AOP对象。AOP的关注点是组成系统的非核心通用服务模块(比如登录检查等),相对于普通对象,aop不需要通过继承、方法调用的方式来提供功能,只需要在xml文件中以引用的方式,将非核心服务功能引用给需要改功能的核心业务逻辑对象或方法中。最终实现对象的解耦。spring 中ioc技术实现了核心业务逻辑对象之间的解耦(如LoginAction与DaoImpl)

AOP可以说是Spring中最难理解的一个知识点了,你可能网上找了很多资料,也买过很多本书,但都不是很理解到底什么是AOP?我曾经也是琢磨了好久才有了一定的了解。那么,到底怎么讲这个知识点呢。来不及解释了,快上车!听完这个例子,我相信你一定会对AOP有一个非常深刻的理解!
让我们新建一个英雄类:

package com.spring.bean;

public class Hero {
    
    private String heroName;
    private String type;
    private String description;
    
    public String getHeroName() {
        return heroName;
    }
    public void setHeroName(String heroName) {
        this.heroName = heroName;
    }
    public String getType() {
        return type;
    }
    public void setType(String type) {
        this.type = type;
    }
    public String getDescription() {
        return description;
    }
    public void setDescription(String description) {
        this.description = description;
    }
    @Override
    public String toString() {
        return "Hero [heroName=" + heroName + ", type=" + type + ", description=" + description + "]";
    }

}

再来个露娜类,继承自英雄类:

image.png

package com.spring.bean;

public class Luna extends Hero{
    
    /**
     * 秀操作
     */
    public void operation(){
        System.out.println("看我月下无限连!");
    }
    
    /**
     * 跑路
     */
    public void run(){
        System.out.println("我操,大空了,赶紧跑!");
    }
    
    /**
     * 发信息
     * @param str
     */
    public void say(String str){
        System.out.println(str);
    }
    
}

可以看到,露娜类有三个方法,分别是秀操作,跑路和发信息。
再写一个团战类:

package com.spring.test;

import com.spring.bean.Luna;

/**
 * 团战类
 * @author Administrator
 *
 */
public class Battle {
    
    public void tuan(){
        Luna luna = new Luna();
        luna.say("上去开团!");
        luna.operation();
        
    }
    
}

测试代码如下:

ApplicationContext context = new ClassPathXmlApplicationContext("spring-aop.xml");
Battle battle = (Battle)context.getBean("Battle");
battle.tuan();

我们用spring把Battle类配上去。

spring-aop.xml

<bean id = 'Battle' class="com.spring.test.Battle"></bean>

运行测试代码:
image.png
image.png

在团战方法里面,我们新建一个露娜的对象,然后发出信息“上去开团”,接着又秀了一把操作。这是一个比较普通的流程。而事实上,露娜可能需要在团战前就提醒我方队友“等我集合打团”,不要人都没齐,队友就无脑往前冲。OK,我们如何通过代码来实现这个过程呢?很显然,这个过程需要在团战方法执行之前就被执行。这就需要AOP面向切面的技术了。

我们需要写一个类,实现MethodBeforeAdvice接口。

/**
 * Notice 定义一个通知打团的信号 - 团战之前
 * @author Administrator
 *
 */
public class BeforeTuanZhan implements MethodBeforeAdvice{

    @Override
    public void before(Method arg0, Object[] arg1, Object arg2) throws Throwable {
        System.out.print(this.getClass().getName() + " -- ");
        Luna luna = new Luna();
        luna.setHeroName("露娜");
        luna.setType("战士/法师");
        luna.setDescription("月光女神");
        luna.say("等我集合打团!");
        
    }
    
}

我们希望这个方法在团战之前就被执行,怎么做呢?没错,就是在XML文件中做如下配置:

<bean id = 'BeforeTuanZhan' class="com.spring.service.BeforeTuanZhan"></bean>

<aop:config>
    
        <!-- 定义所有可供露娜切入的点(方法) -->
        <!-- 原则上只要时机正确,任何团战露娜都可以切进去! -->
        <aop:pointcut expression="execution(* com.spring.test.Battle.*(..))" id="pointcut"/>
        
        <aop:advisor advice-ref="BeforeTuanZhan" pointcut-ref="pointcut"/>
 
        
    </aop:config>

execution(* com.spring.test.Battle.*(..))
这句话的含义就是,返回值为任意,com.spring.test包里面的Battle类,这个类里面所有的方法都需要切入。所谓的切入,就是在方法执行前,执行中,发生异常的时候执行某个其他的方法。执行中用的不多,一般就用另外三种情况。

现在,我们重新执行测试代码:

ApplicationContext context = new ClassPathXmlApplicationContext("spring-aop.xml");
Battle battle = (Battle)context.getBean("Battle");
battle.tuan();

image.png

现在各位想一下,如果团战打赢了怎么办,是不是马上就该去推塔或者打龙啊,这个时候,如果队友团战打赢了就发呆,那就很坑了。所以呢,你这个时候就得提醒队友下一步该做什么,这个提醒的步骤是在团战方法执行结束后才发生的。

我们需要新建一个AfterTuanZhan类,实现AfterReturningAdvice接口。

/**
 * Notice 定义一个团战结束后的类 - 团战之后
 * @author Administrator
 */
public class AfterTuanZhan implements AfterReturningAdvice{

    @Override
    public void afterReturning(Object arg0, Method arg1, Object[] arg2, Object arg3) throws Throwable {
        System.out.print(this.getClass().getName() + " -- ");
        Luna luna = new Luna();
        luna.setHeroName("露娜");
        luna.setType("战士/法师");
        luna.setDescription("月光女神");
        luna.say("进攻敌方防御塔!");
    }
    
}

配置到spring-aop.xml中:

<bean id = 'AfterTuanZhan' class="com.spring.service.AfterTuanZhan"></bean>
<!-- 定义一个切面 -->
    <aop:config>
    
        <!-- 定义所有可供露娜切入的点(方法) -->
        <!-- 原则上只要时机正确,任何团战露娜都可以切进去! -->
        <aop:pointcut expression="execution(* com.spring.test.Battle.*(..))" id="pointcut"/>
        
        <aop:advisor advice-ref="BeforeTuanZhan" pointcut-ref="pointcut"/>
        <aop:advisor advice-ref="AfterTuanZhan" pointcut-ref="pointcut"/>
        
    </aop:config>

现在,我们重新执行测试代码:

ApplicationContext context = new ClassPathXmlApplicationContext("spring-aop.xml");
Battle battle = (Battle)context.getBean("Battle");
battle.tuan();

image.png

再来说说万一团战失利的情况,比如露娜断大了咋办,没错,这个时候就是团战发生了异常,我们在Battle类中手动设置一个异常:

/**
 * 团战类
 * @author Administrator
 *
 */
public class Battle {
    
    public void tuan(){
        Luna luna = new Luna();
        luna.say("上去开团!");
        luna.operation();
        
        int i = 1 / 0 ;
        
    }
    
}

然后,编写TuanZhanException类,实现ThrowsAdvice接口:

/**
 * 定义一个团战异常类,万一出现情况就进入这个类
 * @author Administrator
 *
 */
public class TuanZhanException implements ThrowsAdvice {
    
    //该方法会在露娜团战出现异常后自动执行
    public void afterThrowing(Method method, Object[] args, 
            Object target, Exception ex){
        System.out.print(this.getClass().getName() + " -- ");
        Luna luna = new Luna();
        luna.run();
    }
}

配置到spring-aop,xml:

<bean id = 'TuanZhanException' class="com.spring.service.TuanZhanException"></bean>
<!-- 定义一个切面 -->
    <aop:config>
    
        <!-- 定义所有可供露娜切入的点(方法) -->
        <!-- 原则上只要时机正确,任何团战露娜都可以切进去! -->
        <aop:pointcut expression="execution(* com.spring.test.Battle.*(..))" id="pointcut"/>
        
        <aop:advisor advice-ref="BeforeTuanZhan" pointcut-ref="pointcut"/>
        <aop:advisor advice-ref="AfterTuanZhan" pointcut-ref="pointcut"/>
        <aop:advisor advice-ref="TuanZhanException" pointcut-ref="pointcut"/>
        
    </aop:config>

现在,我们重新执行测试代码:

ApplicationContext context = new ClassPathXmlApplicationContext("spring-aop.xml");
Battle battle = (Battle)context.getBean("Battle");
battle.tuan();

image.png

总结:

1. aop面向切面,切的是什么,没错,切的是方法!
2. 怎么切,你记好了,就是你先自己规定哪些方法需要切,然后设置切入的方式:方法执行之前做什么,执行之后做什么,如果方法出现异常,又要做什么?另外还有一种方法执行的过程中做什么,只是用的比较少,反正我还没有见过在哪里用了。用的最多的就是发生异常后做什么,比如事务管理。

 

posted @ 2017-11-23 08:26  勿忘__初心  阅读(188)  评论(0编辑  收藏  举报