Spring的AspectJ的AOP
来源:互联网 发布:农村淘宝下载官网 编辑:程序博客网 时间:2024/05/19 09:15
AspectJ是一个面向切面的框架,它扩展了java语言。AspectJ定义了专门的AOP语法,所以他有一个编译器用来生成遵守java字节码规范的java文件。
AspectJ的表达式:
语法:execution(表达式)
execution(<访问修饰符>?<返回类型><方法名>(<参数>)<异常>)
- execution(“* com.zhangyike.aspectj.*(..)”) —只检索当前包
- execution(“* com.zhangyike.aspectj..*(..)”) —检索包及当前包的子包.
- execution(* com.zhangyike.aspectj+.*(..)) —检索GenericDAO及子类
AspectJ的增强:
@Before:前置通知,相当与BeforeAdvice
@AfterReturning:后置通知,相当于AfterReturningAdvice
@Around:环绕通知,相当于MethodInterceptor
@AfterThrowing抛出通知,相当于ThrowAdvice
@After:最终通知,不管是否有异常,该通知都会执行
@DeclareParents:引介通知,相当于IntroductionInterceptor。
自动代理的demo
第一步:导包
aspectj依赖aop环境.
第二步:编写目标类
package com.zhangyike.aspectj1;public class UserDao { public void add() { System.out.println("增加用户"); } public void delect() { System.out.println("删除用户"); } public boolean update() { System.out.println("更新用户"); return true; } public void select() { System.out.println("查询用户"); } public void exception(){ System.out.println("异常方法"); int i = 1/0; System.out.println(i); }}
第三步、使用AspectJ注解形式编写增强类
package com.zhangyike.aspectj;import org.aspectj.lang.annotation.Aspect;import org.aspectj.lang.annotation.Before;@Aspect//声明这是一个切点与切面增强的类public class AspectJ1 { //相当于前置增强 @Before(value = "execution(* com.zhangyike.aspectj.UserDao.add(..))") public void before(){ System.out.println("前置增强"); }}
第四步:编写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.xsdhttp://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd"><!-- 创建目标类 --><bean id="userdao" class="com.zhangyike.aspectj.UserDao"></bean><!-- 创建增强类 --><bean id="aspectJ1" class="com.zhangyike.aspectj.AspectJ1"></bean><!-- 自动生成代理方式 ,底层就是AnnotationAwareAspectJAutoProxyCreator--><aop:aspectj-autoproxy/></beans>
第五步:编写测试类
package com.zhangyike.aspectj;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.SpringJUnit4ClassRunner;@RunWith(SpringJUnit4ClassRunner.class)@ContextConfiguration("applicationContext.xml")public class TestDemo { @Autowired @Qualifier("userdao") UserDao userDao; @Test public void test1(){ userDao.add(); System.out.println(); userDao.delect(); System.out.println(); userDao.update(); System.out.println(); userDao.select(); }}
第六步:测试结果
前置增强
增加用户
删除用户
更新用户
查询用户
带有切面的AspectJ代理
第一步:导包
aspectj依赖aop环境.
第二步:编写目标类
package com.zhangyike.aspectj1;public class UserDao { public void add() { System.out.println("增加用户"); } public void delect() { System.out.println("删除用户"); } public boolean update() { System.out.println("更新用户"); return true; } public void select() { System.out.println("查询用户"); } public void exception(){ System.out.println("异常方法"); int i = 1/0; System.out.println(i); }}
第三部:编写增强类
package com.zhangyike.aspectj1;import org.aspectj.lang.ProceedingJoinPoint;/* * 切面类,不用注解声明这是一个切面类,是可以在xml配置这是一个切面类 */public class MyAspectj { public Object around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable{ System.out.println(proceedingJoinPoint.getTarget().getClass().getName() + "环绕增强前---"); //执行方法 Object result = proceedingJoinPoint.proceed(); System.out.println(proceedingJoinPoint.getTarget().getClass().getName() + "环绕增强后---"); return result; } public void before(){ System.out.println(",前置通知!"); } public void afterReturning(Object returnVal){ System.out.println("返回值为:" + returnVal); } public void afterThrowing(Throwable e){ System.out.println("异常通知:" + e.getMessage()); } public void afterFinal(){ System.out.println("最终通知...."); }}
第四步:编写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.xsdhttp://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd"><!-- 创建目标类 --><bean id="userdao" class="com.zhangyike.aspectj1.UserDao"></bean><!-- 创建增强类 --><bean id="aspectJ1" class="com.zhangyike.aspectj1.MyAspectj"></bean><!-- 定义aop的配置 --><aop:config> <!-- 定义切点 --> <aop:pointcut expression="execution(* com.zhangyike.aspectj1.UserDao.add(..))" id="add"/> <aop:pointcut expression="execution(* com.zhangyike.aspectj1.UserDao.delect(..))" id="delect"/> <aop:pointcut expression="execution(* com.zhangyike.aspectj1.UserDao.update(..))" id="update"/> <aop:pointcut expression="execution(* com.zhangyike.aspectj1.UserDao.select(..))" id="select"/> <aop:pointcut expression="execution(* com.zhangyike.aspectj1.UserDao.exception(..))" id="exception"/> <aop:pointcut expression="execution(* com.zhangyike.aspectj1.UserDao.*(..))" id="all"/> <!-- 配置切面 --> <aop:aspect ref="aspectJ1"> <!-- 前置通知的配置 --> <aop:before method="before" pointcut-ref="add"/> <!-- 环绕通知的配置 --> <aop:around method="around" pointcut-ref="delect"/> <!-- 后置通知的配置 --> <aop:after-returning method="afterReturning" pointcut-ref="update" returning="returnVal"/> <!-- 异常通知 --> <aop:after-throwing method="afterThrowing" pointcut-ref="exception" throwing="e"/> <!-- 最终通知 --> <aop:after-returning method="afterFinal" pointcut-ref="all"/> </aop:aspect></aop:config></beans>
第五步:编写测试类
package com.zhangyike.aspectj1;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.SpringJUnit4ClassRunner;@RunWith(SpringJUnit4ClassRunner.class)@ContextConfiguration("applicationContext.xml")public class DemoTest { @Autowired @Qualifier("userdao") UserDao userdao; @Test public void test1(){ userdao.add(); System.out.println(); userdao.delect(); System.out.println(); userdao.update(); System.out.println(); userdao.select(); System.out.println(); userdao.exception(); }}
第六步:测试结果
,前置通知!
增加用户
最终通知….
com.zhangyike.aspectj1.UserDao环绕增强前—
删除用户
com.zhangyike.aspectj1.UserDao环绕增强后—
最终通知….
更新用户
返回值为:true
最终通知….
查询用户
最终通知….
异常方法
异常通知:/ by zero
第七步:执行中出现的异常总结
自动代理类中没有出现异常,只要格式对,基本没什么问题,但是我在编写第二个Demo中出现了好几个异常。
第一个异常:
java.lang.IllegalStateException: Failed to load ApplicationContext
…..
Constructor threw exception; nested exception is java.lang.IllegalArgumentException:
Pointcut is not well-formed: expecting ‘identifier’ at character position 1 (* com.zhangyike.aspectj1.UserDao.add(..))
经过检查发现在xml中切点的格式出错。
第二个错误:
Caused by: org.springframework.beans.BeanInstantiationException: Could not instantiate bean class [org.springframework.aop.aspectj.AspectJPointcutAdvisor]: Constructor threw exception; nested exception is java.lang.IllegalArgumentException: error at ::0 formal unbound in pointcut
at org.springframework.beans.BeanUtils.instantiateClass(BeanUtils.java:163)
at org.springframework.beans.factory.support.SimpleInstantiationStrategy.instantiate(SimpleInstantiationStrategy.java:121)
at org.springframework.beans.factory.support.ConstructorResolver.autowireConstructor(ConstructorResolver.java:280)
… 52 more
Caused by: java.lang.IllegalArgumentException: error at ::0 formal unbound in pointcut
这个错误的原因是在增强类的Before放上有个参数,去掉参数异常就消失了。
五种增强方式中,只有环绕通知和后置通知有参数,其他都没有参数。
- Spring---AspectJ的AOP
- Spring----AspectJ的AOP
- Spring的AspectJ的AOP
- Spring的AspectJ的AOP(*****)
- Spring的AspectJ的AOP
- Spring-AOP:基于AspectJ注解的AOP
- Spring的基于AspectJ的AOP开发
- Spring的aspectj的AOP操作
- 整理的Spring AOP AspectJ切入点语法
- 使用Spring AOP 的@AspectJ记录日志
- Spring AOP支持的AspectJ语法详解
- Spring基于annotation的AOP(AspectJ)
- Spring基于注解@AspectJ的AOP
- Spring基于注解@AspectJ的AOP
- Spring详解------------基于@ASpectJ的AOP
- Spring-AOP切面+Aspectj框架的使用
- Spring的<aop:aspectj-autoproxy>配置
- Spring 的AOP-AspectJ静态实现
- 进程调度算法–时间片轮转调度算法
- 重新认识布局
- Kafka-[2]-Documentation-单机QuickStart
- Java的流
- 为什么数据库读写分离可以提高性能
- Spring的AspectJ的AOP
- 数据访问层4
- 【程序】AVR单片机ATMega16A使用定时器1的输入捕获功能进行红外遥控解码
- 一个java文件里可以有多个类嘛?
- 用js解决杨辉三角问题
- 第十一周总结
- 高精度计算(附高精度加法程序)
- 基本正则表达式个人总结(以CentOS 6.3环境下shell为例)
- POJ 1024 Tester Program 笔记