package com.itheima.aspect;

public class MyAspect {
public void check_Permissions(){
System.out.println("模拟检查权限...");
}
public void log() {
// TODO Auto-generated method stub
System.out.println("记录日志");
}
}

  

package com.itheima.aspect;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy; import com.itheima.jdk.UserDao; public class JdkProxy implements InvocationHandler{
private UserDao userDao;
public Object createProxy(UserDao userDao){
this.userDao=userDao;
ClassLoader classLoader=JdkProxy.class.getClassLoader();
Class[]clazz = userDao.getClass().getInterfaces();
return Proxy.newProxyInstance(classLoader, clazz, this); }
@Override
public Object invoke(Object proxy,Method method,Object[]args)
throws Throwable{
MyAspect myAspect = new MyAspect();
myAspect.check_Permissions();
Object obj = method.invoke(userDao,args);
myAspect.log();
return obj;
}
}

  

package com.itheima.aspect;

import com.itheima.jdk.UserDao;
import com.itheima.jdk.UserDaoImpl; public class JdkTest {
public static void main(String[] args) {
JdkProxy jdkProxy= new JdkProxy();
UserDao userDao = new UserDaoImpl();
UserDao userDao1= (UserDao)jdkProxy.createProxy(userDao);
userDao1.addUser();
userDao1.deleteUser();
}
}

  

AOP动态代理方法的实现如上,不过总体来说还是比较麻烦,要修改的地方也太多。

后来有人开发了aspectJ工具,通过aspectJ,可以方便的使用切面,把代理类和实现类与代理类之间的耦合解除。

aspectJ有两种实现方式,一种是xml,一种是注解。

这里要注意需要导入aspectJ的相关包

方式1:通过xml实现

  将代理类和实现类与代理类之间的调用关系,全部写进xml中:

即以上的代码修改一下:

切面:

  

package com.itheima.aspectj.xml;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint; public class MyAspect {
public void myBefore(JoinPoint joinPoint){
System.out.println("前置通知");
System.out.println("目标类是:"+joinPoint.getTarget());
System.out.println(",被植入增强处理的目标方法位:"+joinPoint.getSignature());
}
public void myAfterReturning(JoinPoint joinPoint){
System.out.println("后置通知");
System.out.println(",被植入增强处理的目标方法位:"+joinPoint.getSignature());
}
public Object myAround(ProceedingJoinPoint joinPoint)
throws Throwable{
System.out.println("环绕开始:执行方法之前,模拟开启事务");
Object object=joinPoint.proceed();
System.out.println("环绕结束:执行方法之后,模拟关闭事务");
return object;
}
public void myAfterThrowing(JoinPoint joinPoint,Throwable e){
System.out.println("异常通知:"+"出错了"+e.getMessage());
System.out.println("目标类是:"+joinPoint.getTarget());
System.out.println(",被植入增强处理的目标方法位:"+joinPoint.getSignature());
}
public void myAfter(JoinPoint joinPoint){
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:context="http://www.springframework.org/schema/context"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc-3.1.1.xsd
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-4.3.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-4.3.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-4.3.xsd">
<bean id = "userDao" class = "com.itheima.jdk.UserDaoImpl"/>
<bean id = "myAspect" class = "com.itheima.aspectj.xml.MyAspect"/>
<aop:config>
<aop:aspect id = "aspect" ref = "myAspect">
<aop:pointcut expression = "execution(* com.itheima.jdk.*.*(..))"
id = "myPointCut"/>
<aop:before method = "myBefore" pointcut-ref="myPointCut"/>
<aop:after-returning method="myAfterReturning"
pointcut-ref = "myPointCut" returning = "returnVal"/>
<aop:around method="myAround" pointcut-ref="myPointCut"/>
<aop:after-throwing method = "myAfterThrowing"
pointcut-ref = "myPointCut" throwing="e"/>
<aop:after method="myAfter" pointcut-ref="myPointCut"/>
</aop:aspect>
</aop:config>
</beans>

测试类:

package com.itheima.aspectj.xml;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import com.itheima.jdk.UserDao;
public class AspectjXmlTest {
public static void main(String[] args) {
ClassPathXmlApplicationContext applicationContext =
new ClassPathXmlApplicationContext("com/itheima/aspectj/xml/applicationContext.xml");
UserDao userDao= (UserDao) applicationContext.getBean("userDao");
userDao.addUser();
userDao.deleteUser();
}
}

  

执行结果:

前置通知
目标类是:com.itheima.jdk.UserDaoImpl@59309333
,被植入增强处理的目标方法位:void com.itheima.jdk.UserDao.addUser()
环绕开始:执行方法之前,模拟开启事务
添加用户
最终通知:模拟方法结束后释放资源
环绕结束:执行方法之后,模拟关闭事务
后置通知
,被植入增强处理的目标方法位:void com.itheima.jdk.UserDao.addUser()
前置通知
目标类是:com.itheima.jdk.UserDaoImpl@59309333
,被植入增强处理的目标方法位:void com.itheima.jdk.UserDao.deleteUser()
环绕开始:执行方法之前,模拟开启事务
删除用户
最终通知:模拟方法结束后释放资源
环绕结束:执行方法之后,模拟关闭事务
后置通知
,被植入增强处理的目标方法位:void com.itheima.jdk.UserDao.deleteUser()

  

aspectJ减少了动态代理实现中出现的代码臃肿,但是,需要修改spring的配置文件,不是很好。

于是衍生了注解方式实现AOP,完美解决了修改spring配置文件的问题。

package com.itheima.aspectj.annotation;

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;
import org.springframework.stereotype.Component;
@Aspect
@Component
public class MyAspect {
//defined the pointcut
@Pointcut("execution(* com.itheima.jdk.*.*(..))")
private void myPointCut(){}
//pre-notification
@Before("myPointCut()")
public void myBefore(JoinPoint joinPoint){
System.out.println("前置通知");
System.out.println("目标类是:"+joinPoint.getTarget());
System.out.println(",被植入增强处理的目标方法位:"+joinPoint.getSignature());
}
//after-notification
@AfterReturning(value="myPointCut()")
public void myAfterReturning(JoinPoint joinPoint){
System.out.println("后置通知");
System.out.println(",被植入增强处理的目标方法位:"+joinPoint.getSignature());
}
//around-notification
@Around("myPointCut()")
public Object myAround(ProceedingJoinPoint joinPoint)
throws Throwable{
System.out.println("环绕开始:执行方法之前,模拟开启事务");
Object object=joinPoint.proceed();
System.out.println("环绕结束:执行方法之后,模拟关闭事务");
return object;
}
@AfterThrowing(value="myPointCut()",throwing="e")
public void myAfterThrowing(JoinPoint joinPoint,Throwable e){
System.out.println("异常通知:"+"出错了"+e.getMessage());
System.out.println("目标类是:"+joinPoint.getTarget());
System.out.println(",被植入增强处理的目标方法位:"+joinPoint.getSignature());
}
@After("myPointCut()")
public void myAfter(JoinPoint joinPoint){
System.out.println("最终通知:模拟方法结束后释放资源");
}
}

  

	<context:component-scan base-package="com.itheima" />
<aop:aspectj-autoproxy />

  

public class AspectjXmlTest {
public static void main(String[] args) {
ClassPathXmlApplicationContext applicationContext =
new ClassPathXmlApplicationContext("com/itheima/aspectj/annotation/applicationContext.xml");
UserDao userDao= (UserDao) applicationContext.getBean("userDao");
userDao.addUser();
userDao.deleteUser();
}
}

  

spring-AOP动态代理,以及aspectJ的xml配置或注解配置方法,各个拦截器的使用顺序的更多相关文章

  1. Spring AOP动态代理原理与实现方式

    AOP:面向切面.面向方面.面向接口是一种横切技术横切技术运用:1.事务管理: (1)数据库事务:(2)编程事务(3)声明事物:Spring AOP-->声明事物   2.日志处理:3.安全验证 ...

  2. spring AOP 动态代理和静态代理以及事务

    AOP(Aspect Oriented Programming),即面向切面编程 AOP技术,它利用一种称为"横切"的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装 ...

  3. Spring AOP 动态代理 缓存

    Spring AOP应用:xml配置及注解实现. 动态代理:jdk.cglib.javassist 缓存应用:高速缓存提供程序ehcache,页面缓存,session缓存 项目地址:https://g ...

  4. spring aop 动态代理批量调用方法实例

    今天项目经理发下任务,需要测试 20 个接口,看看推送和接收数据是否正常.因为对接传输的数据是 xml 格式的字符串,所以我拿现成的数据,先生成推送过去的数据并存储到文本,以便验证数据是否正确,这时候 ...

  5. Spring AOP动态代理

    出现org.springframework.aop.framework.ProxyFactoryBean cannot be cast to 错误 在类型转换的时候, 调用getObject()方法, ...

  6. Spring AOP动态代理实现,解决Spring Boot中无法正常启用JDK动态代理的问题

    Spring AOP底层的动态代理实现有两种方式:一种是JDK动态代理,另一种是CGLib动态代理. JDK动态代理 JDK 1.3版本以后提供了动态代理,允许开发者在运行期创建接口的代理实例,而且只 ...

  7. Spring Aop 动态代理失效分析

    1. Spring Aop 原理 Spring Aop 通过动态代理创建代理对象,在调用代理对象方法前后做增强. 2. Transactional, Async 注解失效? 当在动态代理方法中调用当前 ...

  8. Spring学习笔记之aop动态代理(3)

    Spring学习笔记之aop动态代理(3) 1.0 静态代理模式的缺点: 1.在该系统中有多少的dao就的写多少的proxy,麻烦 2.如果目标接口有方法的改动,则proxy也需要改动. Person ...

  9. spring AOP 之二:@AspectJ注解的3种配置

    @AspectJ相关文章 <spring AOP 之二:@AspectJ注解的3种配置> <spring AOP 之三:使用@AspectJ定义切入点> <spring ...

随机推荐

  1. win7/Ubuntu双系统给Ubuntu扩大硬盘

    问题 电脑上装了windows 7和ubuntu双系统,起初分配硬盘的时候,给ubuntu划分的量较少,如今硬盘空间已不够用,另外挂载分区也不够方便.希望能在不重装系统的情况下,从win7的硬盘中分一 ...

  2. redis 启动配置文件加载报错 service redis does not support chkconfig

    # chkconfig: # description:Redis is a persistent key-value database 网上资料 上面的注释的意思是,redis服务必须在运行级2,3, ...

  3. 移动架构-MVVM框架

    MVVM是Model-View-ViewModel的简写.它本质上就是MVC 的改进版.MVVM 就是将其中的View 的状态和行为抽象化,让我们将视图 UI 和业务逻辑分开 MVVM的优点 可重用性 ...

  4. 使用Cython时遇到的cl.exe的问题

    最近使用cython时,遇到一个问题,报错如下: cl.exe /c /nologo /Ox /W3 /GL /DNDEBUG /MD "-Id:\program files\python3 ...

  5. [学习笔记] Blender 常用工具 移动与旋转,缩放, 变换

    点击上面的移动图标之后,可在X,Y,Z轴移动物体. shift+S 之后,可有更多的移动选项. 旋转:可沿X,Y, Z 进行旋转 缩放 还可输入缩放的具体数值,更精确. 变换 可同时做移动.旋转.缩放 ...

  6. java连接zookeeper实现zookeeper的基本操作

    Java服务端连接Zookeeper,进行节点信息的获取,管理…,整理成一个基本工具, 添加依赖: <dependency> <groupId>org.apache.zooke ...

  7. 第一个vue程序:hello,vlue

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta http ...

  8. py kafka

    # https://github.com/confluentinc/confluent-kafka-python/blob/master/examples/consumer.py #生产者 impor ...

  9. Sonya and Bitwise OR CodeForces - 1004F (线段树,分治)

    大意: 给定序列$a$, 给定整数$x$. 两种操作(1)单点修改 (2)给定区间$[l,r]$,求有多少子区间满足位或和不少于$x$. 假设不带修改. 固定右端点, 合法区间关于左端点单调的. 可以 ...

  10. (七)Redis之持久化之RDB方式

    一.持久化概念 所有的数据都存在内存中,从内存当中同步到硬盘上,这个过程叫做持久化过程. 使用方法: 1. rdb持久化方法:在指定的时间间隔写入硬盘 2.         aof方式:将以日志,记录 ...