Spring中AspectJ的实现AOP(XML+注解)

AOP相关术语如下:

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

切入点的表达式

原:<aop:before method="log" pointcut="execution(public void com.spring.demo08.UserDaoImpl.save())"/>
简:<aop:before method="log" pointcut="execution(* *..*.*DaoImpl.save*(..))"/>
 //切入点表达式
1.execution()     固定的,必须写
2.public          可以忽略
3.void            返回值可以出现* 表示任意返回类型,但不能不写
4.可以使用*代替每层包名,不能不写,简写方式*..*方法,任意位置
5.*DaoImpl
6.方法save*()
7.方法参数:save*(..)任意参数

AOP的通知类型

1. 前置通知
        * 在目标类的方法执行之前执行。
        * 配置文件信息:<aop:after method="before" pointcut-ref="myPointcut3"/>
        * 应用:可以对方法的参数来做校验

    2. 最终通知
        * 在目标类的方法执行之后执行,如果程序出现了异常,最终通知也会执行。
        * 在配置文件中编写具体的配置:<aop:after method="after" pointcut-ref="myPointcut3"/>
        * 应用:例如像释放资源

    3. 后置通知
        * 方法正常执行后的通知。
        * 在配置文件中编写具体的配置:<aop:after-returning method="afterReturning" pointcut-ref="myPointcut2"/>
        * 应用:可以修改方法的返回值

    4. 异常抛出通知
        * 在抛出异常后通知
        * 在配置文件中编写具体的配置:<aop:after-throwing method="afterThorwing" pointcut-ref="myPointcut3"/>
        * 应用:包装异常的信息

    5. 环绕通知
        * 方法的执行前后执行。
        * 在配置文件中编写具体的配置:<aop:around method="around" pointcut-ref="myPointcut2"/>
        * 要注意:目标的方法默认不执行,需要使用ProceedingJoinPoint对来让目标对象的方法执行。

XML方式实现

UserDao,UserDaoImol:

package com.spring.demo08;

public interface UserDao {
    public void save();
    public void update();
}

package com.spring.demo08;

public class UserDaoImpl implements UserDao{

    public void save() {
        System.out.println("保存数据库。。。");
    }
    public void update() {
        System.out.println("更新数据库。。。");
    }

}

切面类:

package com.spring.demo08;

/*
 * 切面类
 */
public class MyAspectXml {
    //通知(具体的增强)
    public void log() {
        System.err.println("日志记录。。。");
    }
}

配置文件配置AOP:

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

    <!-- 配置用户 -->
    <bean id="userDao" class="com.spring.demo08.UserDaoImpl"/>
    <!-- 配置切面类 -->
    <bean id="myAspectXml" class="com.spring.demo08.MyAspectXml"/>
    <!-- 配置AOP -->
    <aop:config>
        <!-- 配置切面类:切入点+通知 -->
        <aop:aspect ref="myAspectXml">
            <!-- 配置的前置通知,save方法执行前,增强方法会执行 -->
            <!-- 切入点的表达式 :execution(public void com.spring.demo08.UserDaoImpl.save())-->
            <!--
                1.execution()     固定的,必须写
                2.public          可以忽略
                3.void            返回值可以出现* 表示任意返回类型,但不能不写
                4.可以使用*代替每层包名,不能不写,简写方式*..*方法,任意位置
                5.*DaoImpl
                6.方法save*()
                7.方法参数:save*(..)任意参数
             -->
            <!-- <aop:before method="log" pointcut="execution(public void com.spring.demo08.UserDaoImpl.save())"/> -->
            <aop:before method="log" pointcut="execution(* *..*.*DaoImpl.save*(..))"/>
        </aop:aspect>
    </aop:config>
</beans>

测试:

package com.spring.demo08;

import javax.annotation.Resource;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

/*
 * 测试AOP功能
 */

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class TestAOP {

    @Resource(name="userDao")
    private UserDao userDao;

    @Test
    public void run() {
        userDao.save();
        userDao.update();
    }
}

运行结果:

AOP实现注解方式

UserDao,UserDaoImpl:

package com.spring.demo09;

public interface UserDao {
    public void save();
    public void update();
}

package com.spring.demo09;

public class UserDaoImpl implements UserDao{

    public void save() {
        System.out.println("保存数据库。。。");
    }
    public void update() {
        System.out.println("更新数据库。。。");
    }

}

切面类:

package com.spring.demo09;

import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;

/*
 * 注解的方式切面类
 */
@Aspect
public class MyAspectAnno {
    /**
     * 切入点:@Before前置通知
     */
    @Before(value="execution(* *..*.*DaoImpl.save*(..))")
    public void log() {
        System.err.println("日志记录。。。");
    }
}

配置:

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

        <!-- 开启自动代理 -->
        <aop:aspectj-autoproxy />
        <!-- 配置目标对象 -->
        <bean id="userDao" class="com.spring.demo09.UserDaoImpl"/>
        <!-- 配置切面类 -->
        <bean id="myAspectAnno" class="com.spring.demo09.MyAspectAnno"/>

</beans>

测试:

package com.spring.demo09;

import javax.annotation.Resource;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

/*
 * 测试AOP功能
 */

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class TestAOP {

    @Resource(name="userDao")
    private UserDao userDao;

    @Test
    public void run() {
        userDao.save();
        userDao.update();
    }
}

运行结果:

通知类型+自定义切入点:

通知类型:

* @Before            -- 前置通知
* @AfterReturing     -- 后置通知
* @Around            -- 环绕通知
* @After             -- 最终通知
* @AfterThrowing     -- 异常抛出通知

package com.spring.demo09;

import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;

/*
 * 注解的方式切面类
 */
@Aspect
public class MyAspectAnno {
    /**
     * 切入点:@Before前置通知
     */
    @Before(value="MyAspectAnno.fn()")
    public void log() {
        System.err.println("日志记录。。。");
    }
    @After(value="MyAspectAnno.fn()")
    public void after() {
        System.err.println("最终通知。。。");
    }
    @Around(value="MyAspectAnno.fn()")
    public void around(ProceedingJoinPoint joinPoint) {
        System.out.println("环绕通知1。。。。");
        try {
            joinPoint.proceed();
        } catch (Throwable e) {
            e.printStackTrace();
        }
        System.out.println("环绕通知2。。。。");
    }

    //自定义切入点
    @Pointcut(value="execution(* *..*.*DaoImpl.save*(..))")
    public void fn() {}
}

运行结果:

原文地址:https://www.cnblogs.com/jumpkin1122/p/11624674.html

时间: 2024-11-09 07:00:34

Spring中AspectJ的实现AOP(XML+注解)的相关文章

Spring AOP报错处理 Can not set field to $Proxy 在spring中使用事物或AOP遇到的错误

[转] 解决方法: http://forum.springsource.org/showthread.php?85016-IllegalArgumentException-with-ApplicationContextAware-Proxy 在配置文件中加入proxy-target-class="true" <tx:annotation-driven transaction-manager="transactionManager" proxy-target-c

Spring中的IOC和AOP

一.IOC(DI) Inversion of Control 控制反转,也叫(Dependency Injection)依赖注入. 我们平常使用对象的时候,一般都是直接使用关键字类new一个对象,那这样有什么坏处呢?其实很显然的,使用new那么就表示当前模块已经不知不觉的和 new的对象耦合了,而我们通常都是更高层次的抽象模块调用底层的实现模块,这样也就产生了模块依赖于具体的实现,这样与我们JAVA中提倡的面向接口面向抽象编程是相冲突的,而且这样做也带来系统的模块架构问题. 很简单的例子,我们在

Spring中管理Bean以及解析XML

Spring是分层的轻量级框架 以IoC(Inverse of Control 反转控制)和AOP(Aspect Oriented Programming 面向切面编程)为核心 应用Spring的好处: 方便解耦,简化开发 Spring就是一个大工厂,可以将所有对象创建和依赖关系维护,交给Spring管理 AOP编程的支持 Spring提供面向切面编程,可以方便的实现对程序进行权限拦截.运行监控等功能 声明式事务的支持 只需要通过配置就可以完成对事务的管理,而无需手动编程 方便集成各种优秀框架

对Spring中的IOC与AOP简单理解(简单的理解,通俗易懂)

IOC与AOP是Spring中的核心:IOC是控制反转,AOP是面向对象编程,IOC采用了工厂设计模式,AOP采用了代理设计模式. 一.IOC IOC是控制反转,而控制反转是一种思想,而DI是依赖注入,是IOC的实现,DI有构造注入.属性注入(设值注入)(基于注解的方式或者基于XML的方式).在项目程序中,频繁的去手动创建对象,new对象使得代码间耦合度太大,而Spring提供了容器,通俗的讲:你无需知道对象的创建过程,只需从其中拿到结果就可以了:把对象的创建以及各个对象间的关系还有对象的销毁交

关于Spring中的&lt;context:annotation-config/&gt;配置(开启注解)

当我们需要使用BeanPostProcessor时,直接在Spring配置文件中定义这些Bean显得比较笨拙,例如: 使用@Autowired注解,必须事先在Spring容器中声明AutowiredAnnotationBeanPostProcessor的Bean: <bean class="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor "/> 使用 @Req

Spring中@Autowired、@Resource和@Inject注解的使用和区别

在使用Spring进行项目开发的时候,会大量使用到自动装配,那自动装配是什么呢?简单来说:Spring 利用依赖注入(DI)功能,完成SpringIOC容器中各个组件之间的依赖关系赋值管理. 下面介绍和总结可以在在Spring使用的三种自动注入的注解.首先回顾一下最初使用xml进行是如何进行注入的. 零.回顾:XML方式注入 使用 property 或者 constructor-arg 属性. <!-- 配置需要被Spring管理的Bean(创建,创建后放在了Spring IOC容器里面)-->

理解Spring中的IOC和AOP

我们是在使用Spring框架的过程中,其实就是为了使用IOC,依赖注入和AOP,面向切面编程,这两个是Spring的灵魂. 主要用到的设计模式有工厂模式和代理模式 IOC就是典型的工厂模式,通过sessionfactory去注入实例. AOP就是典型的代理模式的体现. 代理模式是常用的java设计模式,他的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息.过滤消息.把消息转发给委托类,以及事后处理消息等.代理类与委托类之间通常会存在关联关系,一个代理类的对象与一个委托类的对象关

Spring中aspectJ的应用&mdash;&mdash;接口的监控

一.背景: 1.由于公司最近在拆分之前的项目,打算把项目改为前后端分离的模式,所以我们后端这边都在做接口的开发 2.最近也在和其他的公司对接他们的接口,在对接的过程中发现他们的接口调用都是有日志存储的,所以突发奇想就想用spring的aop做这个东西玩玩(其实好像拦截器也能实现). 二.过程记录 1.在spring的配置文件中增加xsd文件 xmlns:mvc="http://www.springframework.org/schema/mvc" xmlns:xsi="htt

Spring中的scope配置和@Scope注解

Scope,也称作用域,在 Spring IoC 容器是指其创建的 Bean 对象相对于其他 Bean 对象的请求可见范围.在 Spring IoC 容器中具有以下几种作用域:基本作用域(singleton.prototype),Web 作用域(reqeust.session.globalsession),自定义作用域. 1.Spring 的作用域在装配 Bean 时就必须在配置文件中指明,配置方式如下(以 xml 配置文件为例): <!-- 具体的作用域需要在 scope 属性中定义 -->