Spring AOP的理解

7 篇文章 2 订阅

AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。

它利用一种称为"横切"的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其命名为"Aspect",即切面。所谓"切面",简单说就是那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块之间的耦合度,并有利于未来的可操作性和可维护性。

使用"横切"技术,AOP把软件系统分为两个部分:核心关注点横切关注点。业务处理的主要流程是核心关注点,与之关系不大的部分是横切关注点。横切关注点的一个特点是,他们经常发生在核心关注点的多处,而各处基本相似,比如权限认证、日志、事物。AOP的作用在于分离系统中的各种关注点,将核心关注点和横切关注点分离开来。

AOP核心概念

1、横切关注点

对哪些方法进行拦截,拦截后怎么处理,这些关注点称之为横切关注点

2、切面(aspect)

类是对物体特征的抽象,切面就是对横切关注点的抽象

3、连接点(joinpoint)

被拦截到的点,因为Spring只支持方法类型的连接点,所以在Spring中连接点指的就是被拦截到的方法,实际上连接点还可以是字段或者构造器

4、切入点(pointcut)

对连接点进行拦截的定义

5、通知(advice)

所谓通知指的就是指拦截到连接点之后要执行的代码,通知分为前置、后置、异常、最终、环绕通知五类

6、目标对象

代理的目标对象

7、织入(weave)

将切面应用到目标对象并导致代理对象创建的过程

8、引入(introduction)

在不修改代码的前提下,引入可以在运行期为类动态地添加一些方法或字段

在这之前先来看一个例子:

一个IPerson接口

public interface IPerson
{
    public void Do();
}

一个实现类Person.java

public class Person implements IPerson
{
    @Override
    public void Do() {
        System.out.println("Hello");
    }
}

一个main方法测试:

public class test
{
    public static void main(String[] args) {
        IPerson person = new Person();
        person.Do();
    }
}

输出的就是:Hello

这时候如果想要在上面的person.Do();之前或之后做一些其他的事情可以借助一个类——java中代理

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class PersonHandler implements InvocationHandler
{

    private Object obj;

    public PersonHandler(Object obj) {
        this.obj = obj;
    }

    public static <T> T GetInstance(Object obj) {
        PersonHandler handler = new PersonHandler(obj);
        @SuppressWarnings("unchecked")
        T proxy = (T) Proxy.newProxyInstance(obj.getClass().getClassLoader(), 
                obj.getClass().getInterfaces(), handler);
        return proxy;
    }

    /**
     * proxy: 指代我们所代理的那个真实对象
     * method: 指代的是我们所要调用真实对象的某个方法的Method对象 
     * args:指代的是调用真实对象某个方法时接受的参数
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("before");
        Object result = method.invoke(obj, args);
        System.out.println("after");
        return result;
    }
}

先看一下上面的invoke方法,它有三个参数,意思分别在代码的注释里指出。

这里面通过反射调用方法,当我们把person传进来的时候,就会调用Do()方法,在调用的之前和之后分别加了个输出语句。

这时候改一个main方法看一下效果:

public class test
{
    public static void main(String[] args) {    
        IPerson person = PersonHandler.GetInstance(new Person());
        person.Do();
    }
}

运行输出的结果如下:

before
Hello
after

说明已经拦截到person里的方法并可以做一些其他的事情。

现在为了方便,想把IPerson person = PersonHandler.GetInstance(new Person());里的new Person()换成Person.class,这样就不用每次都new一个对象,在上面的PersonHandler类里面加一个构造方法:

public static <T> T GetInstance(Class<?> cls) {
        try {
            Object obj = cls.newInstance();

             return GetInstance(obj);
        }
        catch (InstantiationException | IllegalAccessException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
            return null;
        }
    }

这样就可以使用IPerson person = PersonHandler.GetInstance(Person.class);来创建实例了。

现在,想简化一下:能不能IPerson person = PersonHandler.GetInstance( );直接使用空参的形式来创建实例呢?

当然也会是可以的,可以使用xml来配置(这是不是就是xml里bean的来源呢?)

通过xml里配置的接口和对应的实现类可以实现这种想法。每次将xml里所有的信息解析出来放到集合里,可以在PersonHandler类的构造里面进行遍历判断当前的是哪一个实例。

这种想法有一定的缺陷:如果类越来越多,上百甚至上千个,再一个一个去xml里配置就显得特别的繁琐。

所以,另一种想法:能不能在上面价加个注解就可以实现实例化呢,如:

public class test
{

    public static void main(String[] args) {
        @XXXX
        IPerson person ;
        person.Do();
    }
}

通过上面的注解,就可以实例化对象直接调用Do()方法。通过扫描查找所有带有@XXXX注解的类在某个方法里全部实例化。用到的时候就可以直接使用。
将上面的注解换成@Autowired不就是Spring里的自动装载吗?

绕了这么一大圈,现在回过头来看看AOP就会非常容易了。

现在在Spring里面使用AOP就相当于在PersonHandler类里面的invoke方法,只不过现在不在那个类里面写了。

Spring提供了几种实现AOP的方式:

一、基于XML配置的Spring AOP

二、使用注解配置AOP

三、AspectJ切点函数

四、AspectJ通知注解

五、零配置实现Spring IoC与AOP

这里暂时先介绍一种xml配置的方式:

先定义一个类:

public class TimeHandler
{
    public void printTime()
    {
        System.out.println("CurrentTime = " + System.currentTimeMillis());
    }
}

xml配置(aop.xml):

<?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:tx="http://www.springframework.org/schema/tx"
	xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans-4.2.xsd
        http://www.springframework.org/schema/aop
        http://www.springframework.org/schema/aop/spring-aop-4.2.xsd">

	<bean id="person" class="com.test.impl.Person" />

	<aop:config>
		<aop:aspect id="time" ref="timeHandler" order="1">
			<aop:pointcut id="addAllMethod"
				expression="execution(* com.test.api.Do.*(..))" />
			<aop:before method="printTime" pointcut-ref="addAllMethod" />
			<aop:after method="printTime" pointcut-ref="addAllMethod" />
		</aop:aspect>
	</aop:config>

</beans>

运行的结果就会在调用Do()之前和之后分别打印出时间。

进阶阶段推荐看一下这篇博客:Spring Aop

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Spring AOP(面向切面编程)是Spring框架中的一个重要模块,用于实现横切关注点的模块化。它通过在程序运行期间动态地将代码织入到应用程序的特定位置,从而实现对横切关注点的处理。 在Spring AOP中,通过定义切面(Aspect)和连接点(Join Point)来实现对横切关注点的处理。切面定义了在何处以及如何进行横切关注点的处理,而连接点则表示在应用程序中可以插入切面的位置。 Spring AOP的核心概念是切面、连接点、通知(Advice)、切点(Pointcut)和引入(Introduction): 1. 切面(Aspect):切面是一个模块化的单元,它包含了一组通知和切点。通知定义了在何时、何地以及如何进行横切关注点的处理,而切点定义了在应用程序中哪些连接点可以插入切面。 2. 连接点(Join Point):连接点是在应用程序中可以插入切面的位置。例如,方法调用、方法执行、异常抛出等都可以作为连接点。 3. 通知(Advice):通知定义了在连接点上执行的操作。常见的通知类型有前置通知(Before)、后置通知(After)、返回通知(After Returning)和异常通知(After Throwing)等。 4. 切点(Pointcut):切点是一个表达式,用于定义哪些连接点可以插入切面。通过切点表达式,可以选择性地将切面应用于特定的连接点。 5. 引入(Introduction):引入允许在现有的类中添加新的方法和属性。 Spring AOP的优点在于它能够将横切关注点与业务逻辑分离,提高了代码的可维护性和可重用性。同时,它也提供了一种非侵入式的方式来实现横切关注点的处理,不需要修改原有的代码。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值