SpringAOP的实现原理

什么是AOP

简单的说它就是把我们程序重复的代码抽取出来,在需要执行的时候,使用动态代理的技术,在不修改源码的基础上,对我们的已有方法进行增强。

动态代理

动态代理的常用两种方式

1.基于接口的动态代理
提供者:JDK官方的Proxy类
要求:被代理类最少实现一个接口
2.基于子类的动态代理
提供者:第三方的CGLib。
要求:被代理类不能用final修饰的类(最终类)。

此处我们使用的是一个演员的例子:
在很久以前,演员和剧组都是直接见面联系的。没有中间人环节。 而随着时间的推移,产生了一个新兴职业:经纪人(中间人),这个时候剧组再想找演员就需要通过经纪人来找了。下面我们就用代码演示出来。

基于接口的动态代理
public interface IActor { 
	//基本演出  
	public void basicAct(float money); 
	//危险演出 
	public void dangerAct(float money); 
}
//实现了接口,就表示具有接口中的方法实现。即:符合经纪公司的要求 
public class Actor implements IActor{ 
	public void basicAct(float money){ 
		System.out.println("拿到钱,开始基本的表演:"+money); 
	} 
	public void dangerAct(float money){ 
		System.out.println("拿到钱,开始危险的表演:"+money); 
	} 
}
public class Client { 
	public static void main(String[] args) { 
		//一个剧组找演员: 
		final Actor actor = new Actor();//直接 
		/** 
		* 代理:获取代理对象
		* 要求:被代理类最少实现一个接口 
		* 创建的方式 * Proxy.newProxyInstance(三个参数) 
		* ClassLoader:和被代理对象使用相同的类加载器。 * 
		* Interfaces:和被代理对象具有相同的行为。实现相同的接口。 
		* InvocationHandler:如何代理。
		*/
		IActor proxyActor = (IActor) Proxy.newProxyInstance( actor.getClass().getClassLoader(), actor.getClass().getInterfaces(), new InvocationHandler() {
			/** 
			 * 执行被代理对象的任何方法,都会经过该方法,此方法有拦截的功能 
			 * 参数: 
			 * proxy:代理对象的引用。不一定每次都用得到 * 
			 * method:当前执行的方法对象 * 
			 * args:执行方法所需的参数 * 
			 * 返回值: * 当前执行方法的返回值 */ 
			 @Override 
			 public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { 
				 String name = method.getName(); 
				 Float money = (Float) args[0]; 
				 Object rtValue = null;
				 //每个经纪公司对不同演出收费不一样,此处开始判断 
				 if("basicAct".equals(name)){ 
					 //基本演出,没有2000不演 
					 if(money > 2000){ 
						 //看上去剧组是给了8000,实际到演员手里只有4000 
						 //这就是我们没有修改原来basicAct方法源码,对方法进行了增强 
						 rtValue = method.invoke(actor, money/2); 
					 } 
				 } 
				 if("dangerAct".equals(name)){ 
					 //危险演出,没有5000不演 
					 if(money > 5000){ 
						 //看上去剧组是给了50000,实际到演员手里只有25000 
						 //这就是我们没有修改原来dangerAct方法源码,对方法进行了增强 
						 rtValue = method.invoke(actor, money/2); 
					 } 
				 } 
				 	return rtValue; 
				 	} 
				 }); 
				 //没有经纪公司的时候,直接找演员。 
				 // actor.basicAct(1000f); 
				 // actor.dangerAct(5000f); 
				 //剧组无法直接联系演员,而是由经纪公司找的演员 
				 proxyActor.basicAct(8000f); 
				 proxyActor.dangerAct(50000f); } 
		 }
基于子类的动态代理

还是那个演员的例子,只不过不让他实现接口。

public class Actor{
	//没有实现任何接口 
	public void basicAct(float money){ 
		System.out.println("拿到钱,开始基本的表演:"+money); 
	} 
	public void dangerAct(float money){ 
		System.out.println("拿到钱,开始危险的表演:"+money);
	} 
} 
public class Client { 
	/** 
	* 基于子类的动态代理
	* 要求:被代理对象不能是最终类
	* 用到的类:Enhancer  
	* 用到的方法:create(Class, Callback) 
	* 方法的参数:Class:被代理对象的字节码 Callback:如何代理
	*/ 
	public static void main(String[] args) { 
		final Actor actor = new Actor(); 
		Actor cglibActor = (Actor) Enhancer.create(actor.getClass(), new MethodInterceptor() { 
		/** 
		* 执行被代理对象的任何方法,都会经过该方法。在此方法内部就可以对被代理对象的任何方法进行增强。 
		* 参数: 前三个和基于接口的动态代理是一样的,MethodProxy:当前执行方法的代理对象。 
		* 返回值: 当前执行方法的返回值
		*/ 
		@Override 
		public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable { 
			String name = method.getName(); 
			Float money = (Float) args[0]; 
			Object rtValue = null; 
			if("basicAct".equals(name)){ 
				//基本演出 
				if(money > 2000){ 
					rtValue = method.invoke(actor, money/2); 
				} 
			}
			if("dangerAct".equals(name)){ 
				//危险演出 
				if(money > 5000){ 
					rtValue = method.invoke(actor, money/2); 
				} 
			} 
			return rtValue; 
			} 
		}); 
		cglibActor.basicAct(10000); 
		cglibActor.dangerAct(100000); 
	} 
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值