Spring AspectJ AOP框架注解原理解析

什么是AspectJ

AspectJ是一个面向切面的框架,它扩展了Java语言。AspectJ定义了AOP语法所以它有一个专门的编译器用来生成遵守Java字节编码规范的Class文件。

AspectJ是一个基于Java语言的AOP框架

Spring2.0以后新增了对AspectJ切点表达式支持

@AspectJ 是AspectJ1.5新增功能,通过JDK5注解技术,允许直接在Bean类中定义切面

新版本Spring框架,建议使用AspectJ方式来开发AOP

AspectJ表达式:

语法:execution(表达式)

  • execution(<访问修饰符>?<返回类型><方法名>(<参数>)<异常>)
  • execution(“ cn.itcast.spring3.demo1.dao.(..)”) ---只检索当前包
  • execution(“ cn.itcast.spring3.demo1.dao..(..)”) ---检索包及当前包的子包.
  • execution( cn.itcast.dao.GenericDAO+.(..)) ---检索GenericDAO及子类

AspectJ增强:

  • @Before 前置通知,相当于BeforeAdvice
  • @AfterReturning 后置通知,相当于AfterReturningAdvice
  • @Around 环绕通知,相当于MethodInterceptor
  • @AfterThrowing抛出通知,相当于ThrowAdvice
  • @After 最终final通知,不管是否异常,该通知都会执行
  • @DeclareParents 引介通知,相当于IntroductionInterceptor (不要求掌握)

基于注解

第一步:引入相应jar包.

aspectj依赖aop环境.

spring-aspects-3.2.0.RELEASE.jar

com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar

第二步:编写被增强的类:

UserDao

package cn.spring3.demo1;
/**
 * @author NOP
 * 被代理的对象
 */
public class UserDao {
  public void add() {
    // TODO Auto-generated method stub
    System.out.println("添加客户");
  }
  public void delete() {
    // TODO Auto-generated method stub
    System.out.println("删除客户");
    int i=1/0;
  }
  public String find() {
    // TODO Auto-generated method stub
    System.out.println("查询客户");
    return "fanhuizhi";
  }
  public void update() {
    // TODO Auto-generated method stub
    System.out.println("修改客户");
  }
}

第三步:使用AspectJ注解形式:

package cn.spring3.demo1;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
/**
 * @author NOP 切面类:就是切点与增强结合
 * 前置增强
 */
@Aspect
public class MyAspect {
  @Before(value = "execution(* cn.spring3.demo1.UserDao.add(..))")//这里写表达式,写哪些类需要添加
  public void before(JoinPoint joinpoint) {
    System.out.println("前置增强..."+joinpoint);
  }

  @AfterReturning(value = "execution(* cn.spring3.demo1.UserDao.find(..))",returning="returnVal")//这里写表达式,写哪些类需要添加
  public void afterReturning(Object returnVal){
    System.out.println("后置增强..."+"方法的返回值"+returnVal);
  }

  @Around(value = "execution(* cn.spring3.demo1.UserDao.delete(..))")//这里写表达式,写哪些类需要添加
  public Object around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable{
    System.out.println("环绕前增强...");
    Object obj = proceedingJoinPoint.proceed();
    System.out.println("环绕后增强...");
    return obj;
  }

  @AfterThrowing(value = "execution(* cn.spring3.demo1.UserDao.delete(..))",throwing="ex")//这里写表达式,写哪些类需要添加
  public void afterThrowing(Throwable ex) throws Throwable{
    System.out.println("不跑了出异常了..."+ex.getMessage());
  }

  //@After(value = "execution(* cn.spring3.demo1.UserDao.delete(..))")//这里写表达式,写哪些类需要添加
  @After("MyAspect.MyPointCut()")//类名.方法名
  public void after(){
    System.out.println("最终通知");//不管有没有异常都会通知
  }

  //仅是为了定义一个通用的表达式
  @Pointcut(value = "execution(* cn.spring3.demo1.UserDao.delete(..))")
  private void MyPointCut(){
  }
}

第四步:创建applicationContext.xml

 xmlns:aop="http://www.springframework.org/schema/aop"

* 引入aop的约束:
*<!-- 自动生成代理 底层就是AnnotationAwareAspectJautoProxyCreator -->
  <aop:aspectj-autoproxy/>

  <bean id="userDao" class="cn.spring3.demo1.UserDao"/>

  <bean id="MyAspect" class="cn.spring3.demo1.MyAspect"/>

第五步,编写测试类

package cn.spring3.demo1;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4Cla***unner;

/**
 * @author NOP
 * 自动方式代理没有切点切面的增强
 */
@RunWith(SpringJUnit4Cla***unner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class SpringTest1 {
  @Autowired
  @Qualifier("userDao")
  private UserDao userDao;

  @Test
  public void demo1(){
    System.out.println("-----------------");
    userDao.add();
    System.out.println("-----------------");
    userDao.find();
    System.out.println("-----------------");
    userDao.delete();
    System.out.println("-----------------");
    userDao.update();
    System.out.println("-----------------");
  }
}
测试结果:
-----------------
前置增强...execution(void cn.spring3.demo1.UserDao.add())
添加客户
-----------------
查询客户
后置增强...方法的返回值fanhuizhi
-----------------
环绕前增强...
删除客户
最终通知
不跑了出异常了.../ by zero

AspectJ的通知类型:

  • @Before 前置通知,相当于BeforeAdvice
  • 就在方法之前执行.没有办法阻止目标方法执行的.
  • @AfterReturning 后置通知,相当于AfterReturningAdvice
  • 后置通知,获得方法返回值.
  • @Around 环绕通知,相当于MethodInterceptor
  • 在可以方法之前和之后来执行的,而且可以阻止目标方法的执行.
  • @AfterThrowing抛出通知,相当于ThrowAdvice
  • @After 最终final通知,不管是否异常,该通知都会执行
  • @DeclareParents 引介通知,相当于IntroductionInterceptor (不要求掌握)

切点的定义:

@Pointcut("execution(* cn.itcast.spring3.demo1.UserDao.find(..))")
private void myPointcut(){}

面试:

Advisor和Aspect的区别?

Advisor:Spring传统意义上的切面:支持一个切点和一个通知的组合.

Aspect:可以支持多个切点和多个通知的组合

基于XML

ProductDao

第一步:编写被增强的类:

package cn.spring3.demo2;

public class ProductDao {

  public void add() {
    // TODO Auto-generated method stub
    System.out.println("添加商品");
  }

  public void delete() {
    // TODO Auto-generated method stub
    System.out.println("删除商品");

  }

  public void find() {
    // TODO Auto-generated method stub
    System.out.println("查询商品");
    int i = 1 / 0;
  }

  public String update() {
    // TODO Auto-generated method stub
    System.out.println("修改商品");
    return "woshitest";
  }

}

第二步:定义切面

package cn.spring3.demo2;

import org.aspectj.lang.ProceedingJoinPoint;

/**
 * @author NOP
 * 切面类
 */
public class MyAspectXML {
  /*
   *
   */
  public void before(){
    System.out.println("前置增强");
  }

  public void afterReturning(Object returnVal){
    System.out.println("后置增强"+returnVal);
  }

  public Object around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable{
    System.out.println("环绕前增强");
    Object obj = proceedingJoinPoint.proceed();
    System.out.println("环绕后增强");
    return obj;
  }

  public void afterThrowing(Throwable ex){
    System.out.println("不跑了出异常了..."+ex.getMessage());
  }

  public void after(){
    System.out.println("最终通知");//不管有没有异常都会通知
  }
}

第三步:配置applicationContext.xml

<!-- 定义被增强的类 -->
  <bean id="productDao" class="cn.spring3.demo2.ProductDao"/>

  <!-- 定义切面 -->
  <bean id="myAspectXML" class="cn.spring3.demo2.MyAspectXML"/>

  <!-- 定义aop配置 -->
  <aop:config>
    <!-- 定义切点: -->
    <aop:pointcut expression="execution(* cn.spring3.demo2.ProductDao.add(..))" id="mypointcut"/>

    <!-- 定义切点: -->
    <aop:pointcut expression="execution(* cn.spring3.demo2.ProductDao.update(..))" id="mypointcutar"/>

    <!-- 定义切点: -->
    <aop:pointcut expression="execution(* cn.spring3.demo2.ProductDao.delete(..))" id="mypointcutaar"/>

    <!-- 定义切点: -->
    <aop:pointcut expression="execution(* cn.spring3.demo2.ProductDao.find(..))" id="mypointcutaat"/>

    <aop:aspect ref="myAspectXML">
      <!-- 前置通知-->
      <aop:before method="before" pointcut-ref="mypointcut"/> 

      <!-- 后置通知 -->
      <aop:after-returning method="afterReturning" pointcut-ref="mypointcutar" returning="returnVal"/>

      <!-- 环绕通知 -->
      <aop:around method="around" pointcut-ref="mypointcutaar"/>

      <!-- 异常通知 -->
      <aop:after-throwing method="afterThrowing" throwing="ex" pointcut-ref="mypointcutaat"/>

      <!-- 最终通知//不管有没有异常都会通知 -->
      <aop:after method="after" pointcut-ref="mypointcutaat"/>
    </aop:aspect>
  </aop:config>

第四步:编写测试类

package cn.spring3.demo2;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4Cla***unner;

@RunWith(SpringJUnit4Cla***unner.class)
@ContextConfiguration("classpath:applicationContext2.xml")
public class SpringTest2 {
  @Autowired
  @Qualifier("productDao")
  private ProductDao productDao;

  @Test
  public void demo1(){
    System.out.println("-----------------");
    productDao.add();
    System.out.println("-----------------");
    productDao.find();
    System.out.println("-----------------");
    productDao.delete();
    System.out.println("-----------------");
    productDao.update();
    System.out.println("-----------------");
  }
}

测试结果:
-----------------
前置增强
添加商品
-----------------
查询商品
不跑了出异常了.../ by zero
最终通知

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Spring Aop之AspectJ注解配置实现日志管理的方法

    最近项目要做一个日志功能,我用Spring Aop的注解方式来实现. 创建日志注解 package com.wyj.annotation; import java.lang.annotation.Documented; import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lan

  • Spring用AspectJ开发AOP(基于Annotation)

    基于 Annotation 的声明式 在 Spring 中,尽管使用 XML 配置文件可以实现 AOP 开发,但是如果所有的相关的配置都集中在配置文件中,势必会导致 XML 配置文件过于臃肿,从而给维护和升级带来一定的困难. 为此,AspectJ 框架为 AOP 开发提供了另一种开发方式--基于 Annotation 的声明式.AspectJ 允许使用注解定义切面.切入点和增强处理,而 Spring 框架则可以识别并根据这些注解生成 AOP 代理. 关于 Annotation 注解的介绍如表 1

  • 详解Spring Aop实例之AspectJ注解配置

    上篇<Spring Aop实例之xml配置>中,讲解了xml配置方式,今天来说说AspectJ注解方式去配置spring aop. 依旧采用的jdk代理,接口和实现类代码请参考上篇博文.主要是将Aspect类分享一下: package com.tgb.aop; import org.aspectj.lang.JoinPoint; import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.Aft

  • Spring AOP AspectJ使用及配置过程解析

    这篇文章主要介绍了Spring AOP AspectJ使用及配置过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 AspectJ是一个基于Java语言的AOP框架,Spring2.0以后新增了对AspectJ切点表达式支持.因为Spring1.0的时候Aspectj还未出现; AspectJ1.5中新增了对注解的支持,允许直接在Bean类中定义切面.新版本的Spring框架建 议我们都使用AspectJ方式来开发AOP,并提供了非常灵活且

  • 详解在Spring中如何使用AspectJ来实现AOP

    AspectJ 是通过注解来描述切点与增强的. 1 开发环境要求 因为要使用注解,所以请确保使用的 Java5.0 及以上版本. 引入 AspectJ 相关类库: <dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjrt</artifactId> <version>${aspectj.version}</version> </dependenc

  • Spring使用AspectJ注解和XML配置实现AOP

    本文演示的是Spring中使用AspectJ注解和XML配置两种方式实现AOP 下面是使用AspectJ注解实现AOP的Java Project 首先是位于classpath下的applicationContext.xml文件 <?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmln

  • Spring AspectJ AOP框架注解原理解析

    什么是AspectJ AspectJ是一个面向切面的框架,它扩展了Java语言.AspectJ定义了AOP语法所以它有一个专门的编译器用来生成遵守Java字节编码规范的Class文件. AspectJ是一个基于Java语言的AOP框架 Spring2.0以后新增了对AspectJ切点表达式支持 @AspectJ 是AspectJ1.5新增功能,通过JDK5注解技术,允许直接在Bean类中定义切面 新版本Spring框架,建议使用AspectJ方式来开发AOP AspectJ表达式: 语法:exe

  • Spring Security 安全框架应用原理解析

    Spring Security 简介 背景分析 企业中数据是最重要的资源,对于这些数据而言,有些可以直接匿名访问,有些只能登录以后才能访问,还有一些你登录成功以后,权限不够也不能访问.总之这些规则都是保护系统资源不被破坏的一种手段.几乎每个系统中都需要这样的措施对数据(资源)进行保护.我们通常会通过软件技术对这样业务进行具体的设计和实现.早期没有统一的标准,每个系统都有自己独立的设计实现,但是对于这个业务又是一个共性,后续市场上就基于共享做了具体的落地实现,例如Spring Security,A

  • Spring @Primary和@Qualifier注解原理解析

    一 前言 本篇内容主要是讲解2个重要的注解使用方式和场景,@Primary,@Qualifier注解:其作用就是消除bean注入时的歧义,能够让spring容器知道加载哪个bean: 知识追寻者(Inheriting the spirit of open source, Spreading technology knowledge;) 二 实现方式 如下示例中使用被单接口Sheet, 实现类为SheetA , SHeetB ; 由于注入容器时都是 Sheet类型,会发生异常,此时就是使用@Pri

  • Spring @Conditional注解原理解析

    这篇文章主要介绍了Spring @Conditional注解原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 @Conditional是Spring4新提供的注解,它的作用是根据某个条件加载特定的bean. 我们需要创建实现类来实现Condition接口,这是Condition的源码 public interface Condition { boolean matches(ConditionContext var1, AnnotatedT

  • spring @Component注解原理解析

    这篇文章主要介绍了spring @Component注解原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 1.@controller 控制器(注入服务) 2.@service 业务(注入dao) 3.@repository dao(实现dao访问) 4.@component (把普通pojo实例化到spring容器中,相当于配置文件中的<bean id="" class=""/>) 5.@Comp

  • Spring Boot 文件上传原理解析

    首先我们要知道什么是Spring Boot,这里简单说一下,Spring Boot可以看作是一个框架中的框架--->集成了各种框架,像security.jpa.data.cloud等等,它无须关心配置可以快速启动开发,有兴趣可以了解下自动化配置实现原理,本质上是 spring 4.0的条件化配置实现,深抛下注解,就会看到了. 说Spring Boot 文件上传原理 其实就是Spring MVC,因为这部分工作是Spring MVC做的而不是Spring Boot,那么,SpringMVC又是怎么

  • spring cloud Ribbon用法及原理解析

    这篇文章主要介绍了spring cloud Ribbon用法及原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 简介 这篇文章主要介绍一下ribbon在程序中的基本使用,在这里是单独拿出来写用例测试的,实际生产一般是配置feign一起使用,更加方便开发.同时这里也通过源码来简单分析一下ribbon的基本实现原理. 基本使用 这里使用基于zookeeper注册中心+ribbon的方式实现一个简单的客户端负载均衡案例. 服务提供方 首先是一个

  • springboot @ComponentScan注解原理解析

    这篇文章主要介绍了springboot @ComponentScan注解原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 @ComponentScan 告诉Spring从哪里找到bean. 如果你的其他包都在@SpringBootApplication注解的启动类所在的包及其下级包,则你什么都不用做,SpringBoot会自动帮你把其他包都扫描了. 如果你有一些bean所在的包,不在启动类的包及其下级包,那么你需要手动加上@Compone

  • spring boot jar的启动原理解析

     1.前言 近来有空对公司的open api平台进行了些优化,然后在打出jar包的时候,突然想到以前都是对spring boot使用很熟练,但是从来都不知道spring boot打出的jar的启动原理,然后这回将jar解开了看了下,与想象中确实大不一样,以下就是对解压出来的jar的完整分析. 2.jar的结构 spring boot的应用程序就不贴出来了,一个较简单的demo打出的结构都是类似,另外我采用的spring boot的版本为1.4.1.RELEASE网上有另外一篇文章对spring

  • 关于spring中aop的注解实现方法实例详解

    前言 在之前的一篇文章中我们讲到spring的xml实现,这里我们讲讲使用注解如何实现aop呢.前面已经讲过aop的简单理解了,这里就不在赘述了.话不多说,来一起看看详细的介绍: 注解方式实现aop我们主要分为如下几个步骤: 1.在切面类(为切点服务的类)前用@Aspect注释修饰,声明为一个切面类. 2.用@Pointcut注释声明一个切点,目的是为了告诉切面,谁是它的服务对象.(此注释修饰的方法的方法体为空,不需要写功能比如 public void say(){};就可以了,方法名可以被候命

随机推荐