spring总结(三)

AOP概述:

什么是AOP:

AOP:全称是Aspect Oriented Programming即:面向切面编程。

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

AOP的作用及优势:

作用:

在程序运行期间,不修改源码对已有方法进行增强。

优势:

减少重复代码  提高开发效率  维护方便

AOP的实现方式:

使用动态代理技术

 

动态代理:

动态代理(以下称代理),利用Java的反射技术(Java Reflection),在运行时创建一个实现某些给定接口的新类(也称“动态代理类”)及其实例(对象)

(Using Java Reflection to create dynamic implementations of interfaces at runtime)。

代理的是接口(Interfaces),不是类(Class),更不是抽象类。

 

动态代理的特点:

字节码随用随创建,随用随加载。

它与静态代理的区别也在于此。因为静态代理是字节码一上来就创建好,并完成加载。

装饰者模式就是静态代理的一种体现。

 

动态代理常用的有两种方式:

1、基于接口的动态代理

提供者:JDK官方的Proxy类。

要求:被代理类最少实现一个接口。

2、基于子类的动态代理

提供者:第三方的CGLib,如果报asmxxxx异常,需要导入asm.jar。

要求:被代理类不能用final修饰的类(最终类)。

 

使用JDK官方的Proxy类创建代理对象

此处我们使用的是一个演员的例子:

在很久以前,演员和剧组都是直接见面联系的。没有中间人环节。

而随着时间的推移,产生了一个新兴职业:经纪人(中间人),这个时候剧组再想找演员就需要通过经纪人来找了。下面我们就用代码演示出来。

/**
 * 一个经纪公司的要求:
 * 		能做基本的表演和危险的表演
*/
public interface IActor {
	/**
	 * 基本演出
	 * @param money
	 */
	public void basicAct(float money);
	/**
	 * 危险演出
	 * @param 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);
	}
}

 代理:

              间接。
          获取代理对象:
              要求:
               被代理类最少实现一个接口
          创建的方式
            Proxy.newProxyInstance(三个参数)
          参数含义:
              ClassLoader:和被代理对象使用相同的类加载器。
           Interfaces:和被代理对象具有相同的行为。实现相同的接口。
           InvocationHandler:如何代理。
                   策略模式:使用场景是:
                               数据有了,目的明确。
                               如何达成目标,就是策略。

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);
	}
}

使用CGLib的Enhancer类创建代理对象,还是那个演员的例子,只不过不让他实现接口。


/**
 * 一个演员
*/
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:如何代理
	 * @param args
	 */
	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);
	}
}

解决案例中的问题:思路只有一个:使用动态代理技术创建客户业务层的代理对象,在执行CustomerServiceImpl时,对里面的方法进行增强,加入事务的支持。


/**
 * 用于创建客户业务层对象工厂(当然也可以创建其他业务层对象,只不过我们此处不做那么繁琐)
*/
public class BeanFactory {
	
	/**
	 * 获取客户业务层对象的代理对象
	 * @return
	 */
	public static ICustomerService getCustomerService() {
		//定义客户业务层对象
		final ICustomerService customerService = new CustomerServiceImpl();
		//生成它的代理对象
		ICustomerService proxyCustomerService = (ICustomerService) 
			Proxy.newProxyInstance(customerService.getClass().getClassLoader()
			,customerService.getClass().getInterfaces(), 
			new InvocationHandler() {
			//执行客户业务层任何方法,都会在此处被拦截,我们对那些方法增强,加入事务。	
			@Override
			public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
				String name = method.getName();
				Object rtValue = null;
				try{
					//开启事务
					HibernateUtil.beginTransaction();					
					//执行操作
					rtValue = method.invoke(customerService, args);			
					//提交事务
					HibernateUtil.commit();
				}catch(Exception e){
					//回滚事务
					HibernateUtil.rollback();	
					e.printStackTrace();
				}finally{
					//释放资源.hibernate在我们事务操作(提交/回滚)之后,已经帮我们关了。
					//如果他没关,我们在此处关
				}
				return rtValue;
			}
		});
		return proxyCustomerService;
	}
}

Spring中的AOP

关于代理的选择

在spring中,框架会根据目标类是否实现了接口来决定采用哪种动态代理的方式。

 

 

AOP相关术语

Joinpoint(连接点):

所谓连接点是指那些被拦截到的点。在spring中,这些点指的是方法,因为spring只支持方法类型的连接点。

Pointcut(切入点):

所谓切入点是指我们要对哪些Joinpoint进行拦截的定义。(被增强的方法就是切入点,连接点不一定是切入点,切入点一定是连接点)

Advice(通知/增强):

所谓通知是指拦截到Joinpoint之后所要做的事情就是通知。(就是指要增强的内容

通知的类型:前置通知(指写在切入点方法的前面),后置通知(指写在切入点方法的后面),异常通知(指写在catch方法中),最终通知(指写在finally方法的前面),环绕通知。

Introduction(引介):

引介是一种特殊的通知在不修改类代码的前提下, Introduction可以在运行期为类动态地添加一些方法或Field。

Target(目标对象):

代理的目标对象。(被代理对象)

Weaving(织入):

是指把增强应用到目标对象来创建新的代理对象的过程。

spring采用动态代理织入,而AspectJ采用编译期织入和类装载期织入。

Proxy(代理):(代理)

一个类被AOP织入增强后,就产生一个结果代理类。

Aspect(切面):

是切入点和通知(引介)的结合。

 

 

学习spring中的AOP要明确的事

a、开发阶段(我们做的)

编写核心业务代码(开发主线):大部分程序员来做,要求熟悉业务需求。

把公用代码抽取出来,制作成通知。(开发阶段最后再做):AOP编程人员来做。

在配置文件中,声明切入点与通知间的关系,即切面。:AOP编程人员来做。

b、运行阶段(Spring框架完成的)

Spring框架监控切入点方法的执行。一旦监控到切入点方法被运行,使用代理机制,动态创建目标对象的代理对象,根据通知类别,在代理对象的对应位置,将通知对应的功能织入,完成完整的代码逻辑运行。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值