spring入门2-aop和集成测试
1.AOP开发
1.1.简述
- 作用:面向切面编程:在程序运行期间,在不修改源码的情况下对代码进行增强
- 优势:减少代码重复,提高开发效率,便于维护
- 底层:动态代理实现(jdk动态代理,cglib动态代理)
1.2.术语
Target(目标对象):代理前的对象
Proxy (代理):代理后的增强类
Joinpoint(连接点):指那些被拦截到的点。在spring中,这些点指的是方法,因为spring只支持方法类型的连接点
Pointcut(切入点):被增强的方法
Advice(通知/ 增强):封装增强业务逻辑的方法
Aspect(切面):切点 + 通知
Weaving(织入):将切点与通知结合的过程
1.3.基于xml
1.3.1.快速入门
a.引入aop依赖包
<!-- 里面包含AOP --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>5.0.5.RELEASE</version> </dependency> <!-- aspectj的织入 --> <dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> <version>1.8.13</version> </dependency>
b.创建目标类接口和目标类
interface IUserDao{ void running(); } class UserDao implements IUserDao{ public void running(){ System.out.println("running running"); } }
c.创建切面
class MyAspect{ public void before(){ System.out.println("前置增强"); } }
d.spring配置
<!-- 1.配置目标类和切面类--> <bean id="userDao" class="IUserDao"/> <bean id="myAspect" class="MyAspect"/> <!-- 2.配置切面--> <aop:config> <aop:aspect ref="myAspect"> <aop:before method="before" pointcut="execution(public void UserDao.running())"/> </aop:aspect> </aop:config>
1.3.2.切面配置格式
<aop:config>
<!-- 1.切面配置-->
<aop:aspect ref="切面引用">
<!-- 1.1 配置通知和切点 -->
<aop:通知类型 method="切面类方法名" pointcut="切点表达式" />
<!-- 1.2 配置通知和切点之提取公共切点-->
<aop:pointcut id="公共切点id" expression="切点表达式"></aop:pointcut>
<aop:通知类型 method="切面类方法名" pointcut-ref="公共切点id" />
</aop:aspect>
</aop:config>
1.3.3.通知类型
标签 | 说明 |
---|---|
<aop:before> | 前置通知.指定增强方法在切入点方法之前执行 |
<aop:after-returning> | 后置通知.指定增强方法在切入点方法之后执行 |
<aop:around> | 换绕通知.指定增强方法在切入点方法之前和之后都执行 |
<aop:throwing> | 异常抛出通知.指定增强方法在切入点方法出现异常时执行 |
<aop:after> | 最终通知.指定增强方法在切入点方法执行之后执行,无论是否有异常 |
1.4.切点表达式
- 语法
execution([修饰符] 返回值类型 包名.类名.方法名(参数))
访问修饰符可以省略
返回值类型、包名、类名、方法名可以使用星号* 代表任意
包名与类名之间一个点 . 代表当前包下的类,两个点 .. 表示当前包及其子包下的类
参数列表可以使用两个点 .. 表示任意个数,任意类型的参数列表
- 例子
execution(public void xxx.yyy.aop.Target.method()) execution(void xxx.yyy.aop.Target.*(..)) execution(* xxx.yyy.aop.*.*(..)) execution(* xxx.yyy.aop..*.*(..)) execution(* *..*.*(..))
1.5.基于javaConfig
1.5.1.快速入门
a.创建目标类并交给spring管理
@Component("target")
public class Target implements TargetInterface {
@Override
public void method() {
System.out.println("Target running....");
}
}
@Component("myAspect")
public class MyAspect {
public void before(){
System.out.println("前置代码增强.....");
}
}
b.在切面类中使用注解配置织入关系
@Component("myAspect")
@Aspect
public class MyAspect {
// 提取公共切点
@Pointcut("execution(* com.itheima.aop.*.*(..))")
public void myPoint(){}
// 使用公共切点
@Before("MyAspect.myPoint()")
public void before(){
System.out.println("前置代码增强.....");
}
}
c.在配置中开启aop自动代理和组件扫描
- xml
<!--组件扫描--> <context:component-scan base-package=""/> <!--aop的自动代理--> <aop:aspectj-autoproxy></aop:aspectj-autoproxy>
- javaconfig
@EnableAspectJAutoProxy
1.5.2.注解通知类型
标签 | 说明 |
---|---|
@Before | 前置通知.指定增强方法在切入点方法之前执行 |
@AfterReturning | 后置通知.指定增强方法在切入点方法之后执行 |
@Around | 换绕通知.指定增强方法在切入点方法之前和之后都执行 |
@AfterThrowing | 异常抛出通知.指定增强方法在切入点方法出现异常时执行 |
@After | 最终通知.指定增强方法在切入点方法执行之后执行,无论是否有异常 |
2.集成Junit
2.1.导入test集成包和junit包
<!--此处需要注意的是,spring5 及以上版本要求 junit 的版本必须是 4.12 及以上-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-test</artifactId>
<version>5.0.2.RELEASE</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
<scope>test</scope>
</dependency>
2.2.编写测试代码
// 1.替换运行时
@RunWith(SpringJUnit4ClassRunner.class)
// 2.读取spring配置文件
@ContextConfiguration(classes = {SpringConfigurationz.class}) // javaConfig
// @ContextConfiguration(locations = {"classpath:xxx.xml"}) // xml
public class SpringTest01 {
// 3.注入依赖
@Autowired
private UserDao userDao;
// 4.执行方法测试
@Test
public void t01(){
userDao.xxx.....
}
}