Spring学习之Aspectj开发实现AOP
Aspectj是一个基于Java语言的Aop框架,它提供了强大的Aop功能。
Aspectj简介:
1.Aspectj是一个面向切面的框架,它扩展了Java语言,它定义了一个Aop语法。
2.所以它有一个专门的编译器来生成遵循Java语言的Class文件。
3.Aspectj是一个基于Java语言的Aop框架。
4.在spring2.0之后,加入了对Aspectj切入点表达式的支持。
5.@Aspectj是在jdk1.5之后新增加的注解功能,允许在Bean中直接定义一个切面类
6.新版本的spring框架,建议使用Aspectj方式来开发Aop.
7.使用Aspectj需要导入相关的包:因为Aspectj不是spring的一部分,是和spring一起使用进行Aop操作,spring2.0之后才增加了Aspectj支持。
使用Aspectj来执行aop操作的两种方法:
1.使用xml配置文件来使用Aop操作
2.使用注解来实现Aop操作。
使用Xml配置文件来实现Aop操作的步骤:
1.准备基本的Aop包:(1)基本的spring包;(2)第三方依赖包;(3)有关Aspectj的包;
2.创建spring的核心配置文件,导入Aop的约束:
(1)配置目标类和切面类的bean;
(2)使用<aop:config></aop:config>来创建aop
* <aop:config>
<aop:pointcut expression="execution()" id=""/> //配置切入点
<aop:aspect id="" ref=" 使用哪个切面bean"> //配置切面:把增强用到切入点上的过程
<!--前置通知-->
<aop:before method="使用增强类(切面类)中的哪个方法(方法名)" pointcut-ref=" 使用哪个切入点的id"/>
<!--后置通知-->
<aop:after-returning method="" pointcut-ref=""/>
<!--环绕通知-->
<aop:around method=" " pointcut-ref=" "/>
<!--异常通知-->
<aop:after-throwing method=" " pointcut-ref=" " throwing="e"/>
<!--最终通知-->
<aop:after method=" " pointcut-ref=""/>
</aop:aspect>
</aop:config>
在spring的配置文件中,配置切面使用的是<aop:aspect>元素,该元素会将一个已经定义好的spring Bean转换为切面Bean,所以要现在配置文件中配置好一个普通的spring bean,完成后,使用<aop:aspect>元素的属性ref来引用该bean。
基于xml的Aspectj实现
1.引入相关的包
2.创建一个接口UserDao
package com.itheima.jdk; public interface UserDao {
public void addUser();
public void deleteUser();
}
3.创建一个类实现接口UserDao
package com.itheima.jdk; import org.springframework.stereotype.Repository; public class UserDaoImpl implements UserDao { public void addUser() {
System.out.println("添加用户!"); } public void deleteUser() {
System.out.println("删除用户!"); }
}
4.创建一个切面类
package com.itheima.aspectj.xml; import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint; /**
* 切面类,在此类中编写通知
* @author 12428
*
*/
public class MyAspect {
//前置通知
public void myBefore(JoinPoint joinPoint) { System.out.println("前置通知:执行权限检查!"); //joinPoint.getTarget():获取连接点所在的目标对象
System.out.println("目标类是:"+joinPoint.getTarget()); System.out.println(",被植入曾强的连接点的方法签名:"+joinPoint.getSignature().getName());
} //后置通知
public void myAfterReturning(JoinPoint joinPoint) {
System.out.println("后置通知:模拟记录日志。。。。");
System.out.println("被植入曾强通知目标方法为:"+joinPoint.getSignature().getName());
} //环绕通知
/**
* 环绕通知必须要返回一个Object对象,并且抛出一个错误
* @param proceedingJoinPoint
* @return
* @throws Throwable
*/
public Object myAround(ProceedingJoinPoint proceedingJoinPoint) throws Throwable{ System.out.println("环绕开始:执行目标方法之前:模拟开启事务。。。");
//执行当前的目标方法:proceed()通过反射执行目标方法连接点处的方法
Object object=proceedingJoinPoint.proceed();
System.out.println("环绕结束:执行目标方法之后,模拟关闭事务!"); return object; } //异常通知
public void myAfterThrowing(JoinPoint joinPoint,Throwable e) {
System.out.println("异常通知:"+"出错了"+e.getMessage());
} //最终通知
public void myAfter() {
System.out.println("最终通知:模拟方法结束后的释放资源!");
}
}
5.创建spring配置文件,在com.itheima.aspectj.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-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编程 -->
<aop:config>
<!-- 配置切面 -->
<aop:aspect ref="myAspect">
<!-- 3.1配置切入点:并通知最后增强的方法 -->
<aop:pointcut expression="execution(* com.itheima.jdk.*.*(..))" id="myPointCut"/>
<!-- 3.2关联通知advice和切入点pointcut -->
<!-- 3.2.1前置通知 -->
<aop:before method="myBefore" pointcut-ref="myPointCut"/>
<!-- 3.2.2后置通知,在方法返回之后执行 -->
<aop:after-returning method="myAfterReturning" pointcut-ref="myPointCut"/>
</aop:aspect>
</aop:config>
</beans>
6.测试类,测试
package com.itheima.aspectj.xml; import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext; import com.itheima.jdk.UserDao; /**
* 测试类:用来测试
* @author 12428
*
*/
public class TextMyAspect {
public static void main(String[] args) {
String xmlPath ="com/itheima/aspectj/xml/applicationContext.xml";
//得到容器
ApplicationContext applicationContext=new ClassPathXmlApplicationContext(xmlPath);
//获取UserDao
UserDao userDao=(UserDao) applicationContext.getBean("userDao");
//执行增强后的方法
userDao.addUser();
}
}
Aspectj使用注解来实现AOP
使用基于XML的声明式Aspectj开发主要问题是要配置许多的代码信息,为了解决这个问题,Aspectj框架为Aop提供了一套注解,用以取代spring配置文件中为了实现Aop开发所配置的臃肿代码。
aspectj的注解及其描述
注解名称 | 描叙 |
@Aspect | 用于定义一个切面 |
@PointCut | 用于定义切入点表达式,在使用时要定义一个包含名字和任意参数的方法来表示切入点名称,实际上,就是一个返回值为void,且方法体为空的普通方法 |
@Before |
用于定义一个前置通知,在使用时,通常需要制定一个value的属性值,该属性值用于指定一个切入点表达式(可以时已有的切入点表达式,也可以直接定义切入点表达式) |
@AfterReturning | 用于定义后置通知 |
@Around | 用于定义环绕通知 |
@AfterThrowing | 用于定义异常通知来处理程序中为处理的异常, |
@After | 用于定义最终通知,不管是否异常,该通知都会执行。 |
使用注解来实现AOP操作
1.引入相关的包
2.创建一个接口UserDao
package com.itheima.jdk; public interface UserDao {
public void addUser();
public void deleteUser();
}
3.创建一个类实现接口UserDao
package com.itheima.jdk; import org.springframework.stereotype.Repository; @Repository("userDao")
public class UserDaoImpl implements UserDao { public void addUser() {
System.out.println("添加用户!"); } public void deleteUser() {
System.out.println("删除用户!"); }
}
4.创建切面类
package com.itheima.aspectj.annotation; import org.aopalliance.intercept.Joinpoint;
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; /**
* 设置一个切面类,使用注解来执行
* @author 12428
*
*/
@Aspect //声明这是一个切面类
@Component //声明这是一个Bean
public class MyAspect { //定义一个切入点表达式,并且在下面声明一个方法体为空的
@Pointcut("execution(* com.itheima.jdk.*.*(..))")
private void myPointCut() {} //前置通知
@Before("myPointCut()")
public void myBefore(JoinPoint joinPoint) {
System.out.println("前置通知:模拟执行权限检查..");
System.out.println("目标类是:"+joinPoint.getTarget());
System.out.println("被植入的增强处理的目标方法是:"+joinPoint.getSignature().getName());
} //后置通知
@AfterReturning("myPointCut()")
public void myAfterReturning(JoinPoint joinPoint) {
System.out.println("后置通知:模拟记录日志。。。。");
System.out.println("被植入增强的方法是:"+joinPoint.getSignature().getName());
} //环绕通知
@Around("myPointCut()")
public Object myAround(ProceedingJoinPoint proceedingJoinPoint) throws Throwable{
//开始
System.out.println("环绕通知开始:执行目标方法之前,模拟开启事务。。。");
//执行当前目标方法
Object object=proceedingJoinPoint.getTarget(); //结束
System.out.println("环绕开始:执行目标方法之后,模拟关闭事务。。。");
return object;
} //异常通知
@AfterThrowing(value="myPointCut()",throwing="e")
public void myAfterThrowing(JoinPoint joinPoint,Throwable e) {
System.out.println("异常通知:"+"出错了"+e.getMessage());
} //最终通知
@After("myPointCut()")
public void myAfter() {
System.out.println("最终通知:模拟方法结束后的释放资源!");
} }
首先使用了@Aspect注解定义了一个切面类,因为是需要在spring中使用,因此还需要定义一个@Component注解才能生效。然后使用@PointCut注解来定义一个切入点,并通过定义方法来表示一个切入点的名称。
注意要在目标类UserDaoImpl中添加注解@Respository("UserDao").
5.创建配置文件,在com.itheima.aspectj.annotation
<?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"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-4.3.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-4.3.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-4.3.xsd"> <!-- 指定需要扫描的包,是注解生效 -->
<context:component-scan base-package="com.itheima"/>
<!-- 开启基于注解的声明式Aspectj支持 -->
<aop:aspectj-autoproxy/>
</beans>
6.测试类
package com.itheima.aspectj.annotation; import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext; import com.itheima.jdk.UserDao; //测试类
public class TestAnnotation {
public static void main(String[] args) {
String xmlPath="com/itheima/aspectj/annotation/applicationContext.xml";
ApplicationContext applicationContext=new ClassPathXmlApplicationContext(xmlPath);
//1.从spring容器获得内容
UserDao userDao=(UserDao) applicationContext.getBean("userDao");
userDao.addUser();
}
}
Spring学习之Aspectj开发实现AOP的更多相关文章
- Spring学习--用 ASpectJ 注解实现 AOP
用 AspectJ 注解声明切面: 要在 Spring 中声明 AspectJ 切面 , 只需要在 IOC 容器中将切面声明为 bean 实例.当在 Spring IOC 容器中初始化 AsjectJ ...
- spring 学习(三):aop 学习
spring 学习(三):aop 学习 aop 概念 1 aop:面向切面(方面)编程,扩展功能不修改源代码实现 2 AOP采取横向抽取机制,取代了传统纵向继承体系重复性代码 3 aop底层使用动态代 ...
- Spring学习4-面向切面(AOP)之Spring接口方式
一.初识AOP 关于AOP的学习可以参看帮助文档:spring-3.2.0.M2\docs\reference\html目录下index.html的相关章节 1.AOP:Aspect ...
- Spring学习_day02_AOP,AspectJ,JdbcTemplate
本文为博主辛苦总结,希望自己以后返回来看的时候理解更深刻,也希望可以起到帮助初学者的作用. 转载请注明 出自 : luogg的博客园 谢谢配合! Spring_day02 一.AOP面向切面编程 1. ...
- Spring学习笔记(13)——aop原理及拦截器
原理 AOP(Aspect Oriented Programming),也就是面向方面编程的技术.AOP基于IoC基础,是对OOP的有益补充. AOP将应用系统分为两部分,核心业务逻辑(Core bu ...
- Spring学习笔记-面向切面(AOP)-04
什么是面向切面编程 先大概了解一下部分术语 横切关注点:软件开发中,散布于多出的功能称为横切关注点(cross-cutting concern),简单的可以描述为可以影响应用多处的功能,比如日志.安全 ...
- Spring学习4-面向切面(AOP)之schema配置方式
一.通过Scheme配置实现AOP步骤(Spring AOP环境的环境与上篇博文 Spring接口方式相同) 步骤一.编写业务类: public class AspectBusiness { ...
- Spring学习13-中IOC(工厂模式)和AOP(代理模式)的详细解释
我们是在使用Spring框架的过程中,其实就是为了使用IOC,依赖注入,和AOP,面向切面编程,这两个是Spring的灵魂. 主要用到的设计模式有工厂模式和代理模式. IOC是工厂模式参考:设计模式- ...
- Spring学习笔记(12)——aop
先了解AOP的相关术语:1.通知(Advice):通知定义了切面是什么以及何时使用.描述了切面要完成的工作和何时需要执行这个工作.2.连接点(Joinpoint):程序能够应用通知的一个"时 ...
随机推荐
- 七十八、SAP中数据库操作之查询条数限制
一.UP TO <数量> ROWS,表示查询出多少条数据 二.效果如下
- C#如何编写短信接口,以及接口的调用,包括C#.net访问web,并处理返回值的简例。
在系统的开发中我们经常会用到接口,下面给大家介绍一种短信接口的编写与调用. 我们常调用接口来完成一些信息的通知或者发送验证码,那么这些操作是如何完成的呢?来看一下详细的介绍吧! 首先呢,我们需要有一个 ...
- (五)selenuim和phantonJs处理网页动态加载数据的爬取
selenuim和phantonJs处理网页动态加载数据的爬取 一 图片懒加载 自己理解------就是在打开一个页面的时候,图片数量特别多,图片加载会增加服务器的压力,所以我们在这个时候,就会用到- ...
- 2019-9-16 java上课知识整理总结(动手动脑,课后实验)
java上课知识整理总结(动手动脑,课后实验) 一,课堂测试 1,题目:课堂测试:像二柱子那样,花二十分钟写一个能自动生成30道小学四则运算题目的 “软件” 要求:(1)题目避免重复: (2)可定制( ...
- 在Mac上使用docker+sql server+Navicat
1. 版本: 2. 安装Kubernetes(并不知道安装这个有什么用) git clone https://github.com/maguowei/k8s-docker-desktop-for-m ...
- MyBatis整体架构
Mybatis整体架构 基础支持层 反射模块 Java中的反射很强大,但是还是需要封装的.MyBatis专门提供了反射模块,对元素的反射进行了封装,提供了简洁的API,对反射进行了优化,例如缓存了类的 ...
- ssh and scp从远程服务器下载文件
scp -r root@172.16.252.32:/home/files /home/files 下载目录 -r root是用户172.16.252.32是ip:/home/files 是你要 ...
- 第二阶段scrum-2
1.整个团队的任务量: 2.任务看板: 会议照片: 产品状态: 正在连接配置数据库部分
- iOS延迟执行方法
swift 4.0中dispatch_async,dispatch_after的使用 2018年03月28日 16:15:44 xiao_yuly 阅读数:3576 版权声明:本文为博主原创文章,未经 ...
- 查看 vps 进程网络流量
弄好了 vps 以后,感觉网络流量走的有点多,决定查查看到底什么情况. 首先安装 sar 来看看各个设备消耗的流量 apt-get install sysstat sar 的参数 DEV 表示网口, ...