Spring--基于AspectJ的AOP开发

AspectJ简介

  • ·AspectJ是一个基于Java语言的AOP框架
  • ·Spring2.0以后新增了对AspectJ切点表达式支持
  • ·@AspectJ是AspectJ1.5新增功能,通过JDK5注解技术,允许直接在Bean类中定义切面
  • ·新版本Spring框架,建议使用AspectJ方式来开发AOP
  • ·使用AspectJ需要导入Spring AOP和 Aspect/相关jar包
    • -spring-aop-4.2.4.RELEASE.jar
    • -com.springsource.org.aopalliance-1.0.0.jar
    • -spring-aspects-4.2.4.RELEASE.jar
    • -com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar

基于AspectJ的注解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 definitions here -->

    <!--开启AspectJ的注解开发,自动代理=====================-->
    <aop:aspectj-autoproxy/>

</beans>

@AspectJ提供不同的通知类型

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

在通知中通过value属性定义切点

  • ·通过execution函数,可以定义切点的方法切入
  • ·语法:
    • -execution(<访问修饰符>?<返回类型><方法名>(<参数>)<异常>)
  • ·例如
    • -匹配所有类public方法 execution(public**(.))
    • -匹配指定包下所有类方法 execution(*com.imooc.dao.*(..))不包含子包
    • -execution(*com.imooc.dao..*(..))..*表示包、子孙包下所有类
    • -匹配指定类所有方法 execution(*com.imooc.service.UserService.*(..))
    • -匹配实现特定接口所有类方法
    • execution(*com.imooc.dao.GenericDAO+.*(..))
    • -匹配所有save开头的方法 execution(*save*(..))

  案例(前置通知)

  

    MyAspectAnno

@Aspect
public class MyAspectAnno {
    @Before(value = "execution(* com.windbag.aspectJ.demo1.ProductDao.save(..))")
    public void before(){
        System.out.println("前置通知=====================");
    }

}

    ProductDao

public class ProductDao {

    public void save(){
        System.out.println("保存商品");
    }

    public void update(){
        System.out.println("更改商品");
    }

    public void findone(){
        System.out.println("查询一个商品");
    }

    public void delete(){
        System.out.println("删除商品");
    }

    public void findAll(){
        System.out.println("查询所有商品");
    }
}

    SpringDemo1Test

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

    @Resource(name="productDao")
    private ProductDao productDao;

    @Test
    public void demo1(){
        productDao.delete();
        productDao.findAll();
        productDao.findone();
        productDao.save();
        productDao.update();
    }
}

    applicationContext.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" 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 definitions here -->

    <!--开启AspectJ的注解开发,自动代理=====================-->
    <aop:aspectj-autoproxy/>

    <!--目标类==============================================-->
    <bean id="productDao" class="com.windbag.aspectJ.demo1.ProductDao"/>

    <!--定义切面-->
    <bean class="com.windbag.aspectJ.demo1.MyAspectAnno"/>
</beans>

    其余的通知类型

@Aspect
public class MyAspectAnno {
    @Before(value = "myPointcut1()")
    public void before(){
        System.out.println("前置通知==================");
    }

    @AfterReturning(value = "execution(* com.windbag.aspectJ.demo1.ProductDao.update(..))",returning = "result")
    public void afterReturing(Object result){
        System.out.println("后置通知==================" + result);
    }

    @Around(value = "execution(* com.windbag.aspectJ.demo1.ProductDao.delete(..))")
    public Object around(ProceedingJoinPoint joinPoint) throws Throwable {
        System.out.println("环绕前通知================");
        /*执行目标方法*/
        Object obj = joinPoint.proceed();
        System.out.println("环绕后通知================");
        return obj;
    }

    @AfterThrowing(value = "execution(* com.windbag.aspectJ.demo1.ProductDao.findone(..))",throwing = "e")
    public void afterThrowing(Throwable e){
        System.out.println("异常抛出通知=============" + e.getMessage());
    }

    @After(value = "execution(* com.windbag.aspectJ.demo1.ProductDao.findAll(..))")
    public void after(){
        System.out.println("最终通知================");
    }

    @Pointcut(value = "execution(* com.windbag.aspectJ.demo1.ProductDao.save(..))")
    private void myPointcut1(){

    }
}

基于AspectJ的XML方式的AOP开发

  案例

    CustomerDao

public interface CustomerDao {
    public void save();
    public String update();
    public void delete();
    public void findOne();
    public void findAll();
}

      CustomerImpl

public class CustomerDaoImpl implements CustomerDao {
    @Override
    public void save() {
        System.out.println("保存客户");
    }

    @Override
    public String update() {
        System.out.println("修改客户");
        return "hello";
    }

    @Override
    public void delete() {
        System.out.println("删除客户");
    }

    @Override
    public void findOne() {
        System.out.println("查询一个客户");
        //int a = 1/0;
    }

    @Override
    public void findAll() {
        System.out.println("查询多个客户");
    }
}

      SpringDemo2Test

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext2.xml")
public class SpringDemo2Test {

    @Resource(name="customerDao")
    private CustomerDao customerDao;

    @Test
    public void demo1(){
        customerDao.save();
        customerDao.delete();
        customerDao.findAll();
        customerDao.findOne();
        customerDao.update();
    }
}

    MyAspectXml

public class MyAspectXml {
    //前置通知
    public void before(JoinPoint joinPoint){
        System.out.println("XML方式的前置通知" + joinPoint);
    }
    //后置通知
    public void afterReturing(Object result){
        System.out.println("XML方式的后置通知" + result);
    }
    //环绕通知
    public Object around(ProceedingJoinPoint joinPoint) throws Throwable{
        System.out.println("XML方式的环绕前通知");
        Object obj = joinPoint.proceed();
        System.out.println("XML方式的环绕后通知");
        return obj;
    }
    //异常抛出通知
    public void afterThrowing(Throwable e){
        System.out.println("XML方式的异常抛出通知" + e.getMessage());
    }
    //最终通知
    public void after(){
        System.out.println("XML方式的最终通知");
    }
}

    applicationContext2.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" 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 definitions here -->
    <!--XML的配置方式完成AOP的开发-->
    <!--配置目标类================-->
    <bean id="customerDao" class="com.windbag.aspectJ.demo2.CustomerDaoImpl"/>

    <!--配置切面类-->
    <bean id="myAspectXml" class="com.windbag.aspectJ.demo2.MyAspectXml"/>

    <!--aop的相关配置============-->
    <aop:config>
        <!--配置切入点-->
        <aop:pointcut id="pointcut1" expression="execution(* com.windbag.aspectJ.demo2.CustomerDao.save(..))"/>
        <aop:pointcut id="pointcut2" expression="execution(* com.windbag.aspectJ.demo2.CustomerDao.update(..))"/>
        <aop:pointcut id="pointcut3" expression="execution(* com.windbag.aspectJ.demo2.CustomerDao.delete(..))"/>
        <aop:pointcut id="pointcut4" expression="execution(* com.windbag.aspectJ.demo2.CustomerDao.findOne(..))"/>
        <aop:pointcut id="pointcut5" expression="execution(* com.windbag.aspectJ.demo2.CustomerDao.findAll(..))"/>
        <!--配置AOP的切面-->
        <aop:aspect ref="myAspectXml">
            <!--配置前置增强-->
            <aop:before method="before" pointcut-ref="pointcut1"/>
            <!--配置后置增强-->
            <aop:after-returning method="afterReturing" pointcut-ref="pointcut2" returning="result"/>
            <!--配置环绕增强-->
            <aop:around method="around" pointcut-ref="pointcut3"/>
            <!--配置异常抛出-->
            <aop:after-throwing method="afterThrowing" pointcut-ref="pointcut4" throwing="e"/>
            <!--配置增强-->
            <aop:after method="after" pointcut-ref="pointcut5"/>
        </aop:aspect>

    </aop:config>
</beans>

项目源码

https://github.com/ryanxu47/spring_aspect

原文地址:https://www.cnblogs.com/windbag7/p/9566531.html

时间: 2024-11-11 19:06:55

Spring--基于AspectJ的AOP开发的相关文章

[Spring框架]Spring AOP基础入门总结二:Spring基于AspectJ的AOP的开发.

前言: 在上一篇中: [Spring框架]Spring AOP基础入门总结一. 中 我们已经知道了一个Spring AOP程序是如何开发的, 在这里呢我们将基于AspectJ来进行AOP 的总结和学习. 一, AspectJ的概述: AspectJ是一个面向切面的框架,它扩展了Java语言.AspectJ定义了AOP语法所以它有一个专门的编译器用来生成遵守Java字节编码规范的Class文件. Spring为了简化自身的AOP的开发,将AspectJ拿过来作为Spring自身一个AOP的开发.

(转)Spring使用AspectJ进行AOP的开发:注解方式

http://blog.csdn.net/yerenyuan_pku/article/details/69790950 Spring使用AspectJ进行AOP的开发:注解方式 之前我已讲过Spring使用AspectJ通过配置文件的方式来进行AOP的开发,现在就来讲怎样使用注解方式进行AOP的开发. 创建一个Web项目, 引入相关的jar包.所要导入的jar包如下:  引入Spring的配置文件.主要引入AOP的约束: <?xml version="1.0" encoding=

利用基于@AspectJ的AOP实现权限控制

一. AOP与@AspectJ AOP 是 Aspect Oriented Programming 的缩写,意思是面向方面的编程.我们在系统开发中可以提取出很多共性的东西作为一个 Aspect,可以理解为在系统中,我们需要很多次重复实现的功能.比如计算某个方法运行了多少毫秒,判断用户是不是具有访问权限,用户是否已登录,数据的事务处理,日志记录等等. AOP的术语 连接点(Joinpoint) 程序执行的某个特殊位置:比如类开始初始化前,类初始化后,某个方法调用前,调用后等. 连接点 可 以 理解

Spring的AspectJ的AOP,基于注解(9.1)重点掌握

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

阶段3 2.Spring_08.面向切面编程 AOP_9 spring基于注解的AOP配置

复制依赖和改jar包方式 src下的都复制过来. 复制到新项目里了 bean.xml里面复制上面一行代码到下面.把aop改成context. 配置spring容器创建时要扫描的包 Service的配置这里就可以删除了 配置注解 使用@Service注解 开始AOP配置 把通知类交给Spring来管理 在Logger上加注解.之类注意,@Service和@Repository都不合适.因为logger属于三层 所以这里用@Component这个注解来配置 写完上面的@Component的注解后.b

开涛spring3(6.4) - AOP 之 6.4 基于@AspectJ的AOP

Spring除了支持Schema方式配置AOP,还支持注解方式:使用@AspectJ风格的切面声明. 6.4.1  启用对@AspectJ的支持 Spring默认不支持@AspectJ风格的切面声明,为了支持需要使用如下配置: <aop:aspectj-autoproxy/> 这样Spring就能发现@AspectJ风格的切面并且将切面应用到目标对象. 6.4.2  声明切面 @AspectJ风格的声明切面非常简单,使用@Aspect注解进行声明: @Aspect() Public class

基于aspectj实现AOP操作的两种方式——xml配置

1. 要导入的 jar 包: 常用的aspectj表达式: 权限修饰符可以省略,以下表示:返回值类型为任意,com.chy.service包以及其子包下的.任意类的.参数任意的.任意方法 execution(* com.chy.service..*(..) 2. 在spring的核心配置文件中: 总结: 1. 配置切入点 2. 配置切面:把哪个增强类的哪个方法,前置增强到哪个切入点上 原文地址:https://www.cnblogs.com/cn-chy-com/p/9256048.html

Spring 基于Aspectj切面表达式(6)

1 package com.proc; 2 3 import org.aspectj.lang.JoinPoint; 4 import org.aspectj.lang.ProceedingJoinPoint; 5 import org.aspectj.lang.annotation.After; 6 import org.aspectj.lang.annotation.AfterReturning; 7 import org.aspectj.lang.annotation.AfterThrow

基于aspectj的aop注解操作