浅析SpringAOP的实现及执行过程

SpringAOP的实现及执行过程

为什么实现SpringAOP

例子:
先看如下CalculatorService 类中的代码,实现了ICalculatorService 接口中的抽象方法

public class CalculatorService implements ICalculatorService {

	@Override
	public int mul(int a, int b) {
		System.out.println(this.getClass().getName()+":The mul method begins.");
		System.out.println(this.getClass().getName()+":Parameters of the mul method: ["+a+","+b+"]");
		int result = a*b;
		System.out.println(this.getClass().getName()+":Result of the mul method:"+result);
		System.out.println(this.getClass().getName()+":The mul method ends.");
		return result;
	}

	@Override
	public int div(int a, int b) {
		System.out.println(this.getClass().getName()+":The div method begins.");
		System.out.println(this.getClass().getName()+":Parameters of the div method: ["+a+","+b+"]");
		int result = a/b;
		System.out.println(this.getClass().getName()+":Result of the div method:"+result);
		System.out.println(this.getClass().getName()+":The div method ends.");
		return result;
	}
}

Test中代码

package com.jd.test;

import com.jd.calculator.CalculatorService;
import com.jd.calculator.ICalculatorService;

public class Test {

	public static void main(String[] args) {
		ICalculatorService calculatorService = new CalculatorService();
		int result = calculatorService.mul(1, 1);
		System.out.println("-->"+result);
	}
}

我们可以看见,上边CalculatorService 类中方法体中的代码出现重复,怎么解决代码重复就是接下来要说的SpringAOP。即如何通过Spring AOP思想提高代码的重用性?

AOP

概念:
AOP(Aspect Oriented Programming 面向切面编程)是一种通过运行期动态代理实现代码复用的机制,是对传统OOP(Object Oriented Programming,面向对象编程 )的补充。目前,Aspectj是Java社区里最完整最流行的AOP框架,在Spring 2.0以上版本中可以通过Aspectj注解或基于XML配置AOP。
下边叙述用注解的方法实现SpringAOP
第一步:创建java工程——创建lib文件夹——导入以下jar包
在这里插入图片描述
第二步:创建如下项目结构
在这里插入图片描述
注意:
a、从Spring 3.2开始spring-core-xxx.jar包已集成CGLib和ASM 相关jar包,所以Spring工程不需再额外引入这些jar包,如下图:
在这里插入图片描述
b、即使proxy-target-class设置为false,如果目标类没有声明接口,则Spring将自动使用CGLib生成代理对象。
第三步 :看如下代码,并分析
1,配置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:aop="http://www.springframework.org/schema/aop"
	xsi:schemaLocation="http://www.springframework.org/schema/beans
	 http://www.springframework.org/schema/beans/spring-beans.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">

	<context:component-scan base-package="com.jd"></context:component-scan>
	<!-- 该代码执行,扫描com.jd包及其子包 -->
	
	<!-- proxy-target-class标签属性值为false,则默认为jdk动态代理,获取类的方法为用接口。
	该行代码执行过称为(autoproxy含义): 
	1,Spring获取加了@Aspect注解的类,并获取加了此注解的类中的所有方法,得到其中方法的注解,再得到注解中的表达式。
	2,得到注解表达式后,扫描可以扫描到的所有的类,扫描其中的方法,找到与加了@Aspect注解的类中的方法匹配的方法,获取匹配到的方法存在类,spring自动为该类创建对象。-->
	<aop:aspectj-autoproxy proxy-target-class="false"></aop:aspectj-autoproxy>

</beans>

2,CalculatorAspect类:自定义一个@Aspect修饰的切面类——>将其创建的对象保存于Spring IOC容器——>自定义增强方法(增强方法也称为通知方法,指标注有@Before、@AfterRunning、@AfterThrowing、@After或@Around注解的Java方法。

package com.jd.calculator;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.stereotype.Component;

@Aspect	//切面:由切点(方法)组成。
@Component
public class CalculatorAspect {

	@Before("execution(int mul(int, int))")	//前置增强:在执行方法前执行该方法代码。execution(int mul(..))
	public void before(JoinPoint jp) {
		Object object=jp.getTarget();	//得到代理类
		Object [] args=jp.getArgs();	//获取参数
		String name=jp.getSignature().getName();	//得到方法名
		System.out.println(this.getClass().getName()+":The "+name+" method begins.");
		System.out.println(this.getClass().getName()+":Parameters of the "+name+" method: ["+args[0]+","+args[1]+"]");
	}
}

3,CalculatorService 中代码(以乘法为例说明)

package com.jd.calculator;

import org.springframework.stereotype.Service;

@Service
public class CalculatorService implements ICalculatorService {

	@Override
	public int mul(int a, int b) {
		int result = a*b;	
		System.out.println(this.getClass().getName()+":Result of the mul method:"+result);	
		System.out.println(this.getClass().getName()+":The mul method ends.");	
		return result;
	}

	@Override
	public int div(int a, int b) {
		System.out.println(this.getClass().getName()+":The div method begins.");
		System.out.println(this.getClass().getName()+":Parameters of the div method: ["+a+","+b+"]");
		int result = a/b;
		System.out.println(this.getClass().getName()+":Result of the div method:"+result);
		System.out.println(this.getClass().getName()+":The div method ends.");
		return result;
	}
}

4,Test类

package com.jd.test;

import org.springframework.context.support.ClassPathXmlApplicationContext;

import com.jd.calculator.ICalculatorService;

public class Test {

	public static void main(String[] args) {
	
		//application.xml中配置后。
		ClassPathXmlApplicationContext applicationContext=new ClassPathXmlApplicationContext("application.xml");//获得IOP容器,创建对象,立即执行application.xml中的代码,所以此行代码执行完毕已经得到动态代理对象。
		ICalculatorService calculatorService=applicationContext.getBean(ICalculatorService.class);	//得到对象。此时为jdk代理,此处用接口类获取对象。CGLIB代理用CalculatorService.class也可以。
		System.out.println(calculatorService.getClass().getName());	//执行此行代码之前已经产生了动态代理,所有此处有输出
		int result=calculatorService.mul(1, 7);
		System.out.println("---->"+result);
	}
}

执行上述Test中的代码得到结果为
在这里插入图片描述
代码执行过程:
执行main方法(主方法为程序的入口)——main方法中第一行代码执行,获取xml文件中的信息,找到注解,找匹配的方法,匹配到CalculatorService 中的mul方法;main方法中int result=calculatorService.mul(1, 7);代码执行,开始执行前置增强方法,然后再执行实现类中的mul方法,得到结果输出。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值