`
1028826685
  • 浏览: 938420 次
  • 性别: Icon_minigender_1
  • 来自: 重庆
社区版块
存档分类

基于注解方式声明切面(AOP)

阅读更多
基础知识:


首先启动对@AspectJ注解的支持(蓝色部分):
<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-2.5.xsd
           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd">
<aop:aspectj-autoproxy/>
<bean id="orderservice" class="cn.itcast.service.OrderServiceBean"/>
<bean id="log" class="cn.itcast.service.LogPrint"/>
</beans>

@Aspect
public class LogPrint {
@Pointcut("execution(* cn.itcast.service..*.*(..))")
private void anyMethod() {}//声明一个切入点
@Before("anyMethod() && args(userName)")//定义前置通知
public void doAccessCheck(String userName) {
}
@AfterReturning(pointcut="anyMethod()",returning="revalue")//定义后置通知
public void doReturnCheck(String revalue) {
}
@AfterThrowing(pointcut="anyMethod()", throwing="ex")//定义例外通知
    public void doExceptionAction(Exception ex) {
}
@After("anyMethod()")//定义最终通知
public void doReleaseAction() {
}
@Around("anyMethod()")//环绕通知
public Object doBasicProfiling(ProceedingJoinPoint pjp) throws Throwable {
return pjp.proceed();
}
}



照样实现步骤如下:

第一步:导入需要的*.jar



dist\spring.jar
lib\jakarta-commons\commons-logging.jar
如果使用了切面编程(AOP),还需要下列jar文件
lib/aspectj/aspectjweaver.jar和aspectjrt.jar
lib/cglib/cglib-nodep-2.1_3.jar



第二步:根据上面的基础知识导入命名空间:
       xmlns:aop="http://www.springframework.org/schema/aop"
      http://www.springframework.org/schema/aop   http://www.springframework.org/schema/aop/spring-aop-2.5.xsd"
//打开spring aop
<aop:aspectj-autoproxy/>


第三步:编写我们交个spring管理的拦截器类

@Aspect//声明切面
public class MyItercepter {

//下面是AOP表达式
@Pointcut("execution (* com.liyong.serviceBean.Imp.PersonServiceBean.*(..))")

private void anyMethod() {}//声明一个切入点这是切入点名称

//下面的前置通知表达式意思是满足anyMethod()切入点并且调用函数参数类型是String
@Before("anyMethod() && args(name)")
public void doAccessCheck(String name) {
//在执行拦截方法前调用可得到输入参数
System.out.println("name : "+name);
System.out.println("exctution 前置通知");
}
//得到返回的结果 returning 是调用方法返回的结果作为doReturnCheck()函数的输入参数
@AfterReturning(pointcut="anyMethod()",returning="result")
public void doReturnCheck(String result) {
//在执行拦截方法后调用可得到返回参数
System.out.println("exctution 后置通知");
System.out.println("result : "+result);
}

@AfterThrowing(pointcut="anyMethod()",throwing="e")
public void doExceptionAction(Exception e) {
System.out.println("e : "+e);
System.out.println("exctution 异常通知");
}

@After("anyMethod()")
public void doReleaseAction() {
System.out.println("exctution 最终通知");
}

@Around("anyMethod()")
public Object doBasicProfiling(ProceedingJoinPoint pjp) throws Throwable {
System.out.println("exctution 开始环绕测试");
//必须调用下面的方法
Object object=pjp.proceed();
System.out.println("exctution 结束环绕测试");
return object;
}


}

第四步 :编写业务类和接口

IPersonServiceBean。java PersonServiceBean。java

public interface IPersonServiceBean {

public abstract void save(String name);

public abstract String update(String name);

}

public class PersonServiceBean implements IPersonServiceBean {

public void save(String name)
{
// throw new IllegalArgumentException("抛出异常");
System.out.println("save is invoke");
}

public String update(String name)
{

System.out.println("update is invoke");
return "Sueccess";
}
}

第五步:在这里我们使用xml文件的形式来办bean交个spring管理(还有可以通过类路径扫描让spring管理bean见上面的博客)


<?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:context="http://www.springframework.org/schema/context"
       xmlns:aop="http://www.springframework.org/schema/aop"     
       xsi:schemaLocation="http://www.springframework.org/schema/beans
           http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
           http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd
           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd">
        <aop:aspectj-autoproxy/>

//下面是我们交个spring管理的 bean
     <bean id="personService" class="com.liyong.serviceBean.Imp.PersonServiceBean"/>
<bean id="myItercepter" class="com.liyong.Itecepter.MyItercepter"/>

</beans>

第六步:编写单元测试

@Test
public void TestAOP()
{
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
IPersonServiceBean personServiceBean =(IPersonServiceBean)context.getBean("personService");
//personServiceBean.save("liyong");
personServiceBean.update("xxx");
}

总结:

这里面最重要和核心的就是怎样编写我们的拦截器类代码然后交个spring管理就OK啦
知识讲解:

AOP表达式:

对某个包及类作拦截:

   @Pointcut("execution (* com.liyong.serviceBean.Imp.PersonServiceBean.*(..))")
private void anyMethod() {}//声明一个切入点这是切入点名称

在上面的切入点execution()方法中第一个*表示这个切入点函数返回的类型为任意类型
com.liyong.serviceBean.Imp表示包.PersonServiceBean表示这个包下的类
其中我们也可以这么写com.liyong.serviceBean.Imp.*表示这个包及其子包
.*(..)表示这个类中的所有方法..表示参数任意类型和个数




@Before("anyMethod() && args(name)")
public void doAccessCheck(String name) {
//在执行拦截方法前调用可得到输入参数
System.out.println("name : "+name);
System.out.println("exctution 前置通知");
}
@Before表示前置通知 anyMethod()表示这个切入点的名称类似函数

&& args(name)表示不仅满足anyMethod()这个切入点还满足这个函数参数类型是String





//得到返回的结果 returning 是调用方法返回的结果作为doReturnCheck()函数的输入参数
@AfterReturning(pointcut="anyMethod()",returning="result")
public void doReturnCheck(String result) {
//在执行拦截方法后调用可得到返回参数
System.out.println("exctution 后置通知");
System.out.println("result : "+result);
}





例外通知:
第一个指明切入点是:anyMethod()
第二个参数:throwing="e" 表示这个函数抛出一个异常 作为doExceptionAction(..)
函数的输入值
@AfterThrowing(pointcut="anyMethod()",throwing="e")
public void doExceptionAction(Exception e) {
System.out.println("e : "+e);
System.out.println("exctution 异常通知");
}





最终通知:

anyMethod():切入点名称
@After("anyMethod()")
public void doReleaseAction() {
System.out.println("exctution 最终通知");
}



环绕通知:

@Around("anyMethod()")
public Object doBasicProfiling(ProceedingJoinPoint pjp) throws Throwable {
System.out.println("exctution 开始环绕测试");
//必须调用下面的方法
Object object=pjp.proceed();
System.out.println("exctution 结束环绕测试");
return object;
}



总结各个通知执行顺序:

第一种:没有环绕通知

当某个方法被拦截到了先执行->前置通知->执行这个拦截函数->(如果出现例外就执行
例外通知)->最终通知

第二种:
<1、有环绕通知没有异常情况下
当某个方法被拦截到了先执行->前置通知->执行到环绕通知里的方法
Object object=pjp.proceed();前面的代码
调用这个函数pjp.proceed();
然后去执行这个拦截的方法
然后执行后置通知
最终通知
然后执行Object object=pjp.proceed();后面的代码

<2、有环绕通知有异常情况下

当某个方法被拦截到了先执行->前置通知->执行到环绕通知里的方法
Object object=pjp.proceed();前面的代码
调用这个函数pjp.proceed();
然后去执行这个拦截的方法
有异常执行这个异常通知
然后执行最终通知
注意:
Object object=pjp.proceed();后面的代码不会被执行

代码见附件。。。。。。。。。。。。。。。。。。。。
分享到:
评论

相关推荐

    最简单的基于注解进行面向切面AOP开发案例

    本项目是提供给java新手...本项目演示了java的 AOP 面向切面 的开发。 本项目演示了 前置切面,环绕切面, 后置异常切面 的开发。 本项目的技术栈是 Maven+Spring 如果您是技术高手,请不要购买,这个资源不适合您。

    以注解方式模拟Spring IoC AOP

    - **基于注解的AOP**:Spring支持在方法上直接定义切面注解,如`@Before`, `@After`, `@Around`等。当目标方法被调用时,Spring会检查是否有相关的切面注解并执行相应操作。 为了模拟AOP,我们可以创建一个拦截器...

    注解方式实现AOP编程

    在实际应用中,我们需要将切面类注册到Spring容器中,这通常通过`@Component`注解完成,然后通过`@EnableAspectJAutoProxy`注解启用基于注解的AOP代理。 ```java @Configuration @EnableAspectJAutoProxy public ...

    Spring 基于注解方式实现AOP

    在Spring框架中,基于注解的AOP(面向切面编程)是一种强大的工具,它允许开发者无需编写XML配置即可实现切面。这种编程方式极大地提高了代码的可读性和可维护性。下面我们将深入探讨如何使用注解来实现Spring AOP。...

    使用Spring的注解方式实现AOP的细节

    在Spring中,我们可以使用注解来声明切面、切点和通知。下面将详细介绍这些关键组件: 1. **@Aspect**: 这个注解用于标记一个类作为切面,这个类通常会包含切点和通知。 ```java @Aspect public class ...

    面向切面 aop

    例如,我们可以使用`@Aspect`注解声明一个切面类,`@Before`、`@After`、`@Around`等注解定义通知,`@Pointcut`定义切点。Spring还提供了一个名为`@Profile`的注解,用于在特定环境下启用或禁用切面。 ...

    Spring框架(基于xml、基于注解、AOP面相切面、事务)思维导图

    Spring框架(基于xml、基于注解、AOP面相切面、事务)思维导图

    Spring AOP面向切面三种实现

    Spring AOP提供了三种实现方式:代理模式、AspectJ切面和注解驱动的切面。 1. **代理模式** Spring AOP的基础是动态代理,它可以创建目标对象的代理对象来实现切面功能。有两种代理方式:JDK动态代理和CGLIB代理。...

    spring切面AOP所使用的jar包

    Spring AOP基于代理模型,而AspectJ则是一个完整的AOP编译器,可以在编译时或运行时进行织入。Spring AOP的切入点表达式语法相对较简单,而AspectJ的切入点表达式更强大且灵活。 5. **使用步骤**: - 配置Spring ...

    面向切面编程aop简介

    重点在于理解切面、连接点、切入点和通知的概念,以及如何通过注解或XML配置实现AOP。在实际开发中,常见的应用场景包括日志记录、事务控制、权限验证和缓存管理等。掌握好Spring AOP可以显著提升代码的整洁度和复用...

    Spring基于注解实现AOP

    Spring AOP(面向切面编程)提供了一种优雅的方式来处理系统中的横切关注点,如日志、事务管理等。本篇文章将深入探讨如何在Spring中通过注解实现AOP。 首先,了解AOP的基本概念。面向切面编程是一种编程范式,它...

    基于注解配置和使用spring AOP(spring mvc框架)

    Spring AOP(面向切面编程)是Spring框架的重要组成部分,它提供了一种在不修改源代码的情况下,通过插入额外的代码(称为切面)来增强应用程序功能的方式。在Spring MVC框架中,AOP通常用于实现日志记录、事务管理...

    Spring切面AOP编程的简单模拟实现

    在Spring框架中,AOP(面向切面编程)是一种强大的设计模式,它允许开发者将关注点从核心业务逻辑中分离出来,例如日志记录、事务管理等。本教程将通过模拟Spring AOP来阐述如何实现一个简单的切面编程。我们将讨论...

    aop切面拦截单个方法实例

    自Spring 2.5起,Spring引入了基于注解的AOP,这使得在代码中直接定义切面变得简单。开发者可以使用`@Aspect`注解定义一个切面类,`@Before`、`@After`、`@Around`、`@AfterReturning`和`@AfterThrowing`注解来定义...

    Java 和 Kotlin的注解以及切面实现方式

    - **Spring AOP**:通过定义切面(Aspect),声明通知(Advice),以及指定切入点(Pointcut)来实现AOP。例如: ```java @Aspect public class LoggingAspect { @Before("execution(* com.example.service.*.*...

    Spring mvc mybatis plus 实现AOP 切面日志系统

    在IT行业中,Spring MVC、MyBatis Plus以及AOP(面向切面编程)是Java Web开发中的重要组件,常用于构建高效、灵活的企业级应用。本项目“Spring MVC Mybatis Plus 实现AOP 切面日志系统”旨在提供一个基础的日志...

    用注解的方式进行SpringAOP开发

    在Spring框架中,AOP(面向切面编程)是一种强大的设计模式,它允许开发者将关注点分离,将横切关注点(如日志、事务管理)从核心业务逻辑中解耦出来。本篇我们将深入探讨如何使用注解的方式来实现Spring AOP开发。 ...

    spring注解方式实现aop

    在Spring中,AOP主要分为两种实现方式:基于XML配置和基于注解。本示例主要探讨注解方式。 1. **定义切面(Aspect)** 切面是关注点的模块化,它包含通知(Advice)和切入点(Pointcut)。在Spring中,我们可以...

    使用注解实现AOP

    总结来说,使用注解实现的Spring AOP提供了一种简洁、直观的方式来定义和应用切面。通过注解,我们可以轻松地插入日志、事务管理和安全检查等横切关注点,而无需侵入核心业务代码。这使得代码更加整洁,维护性更强,...

Global site tag (gtag.js) - Google Analytics