用到了AOP的动态代理模式:
1.画图:把切面画出来
2.举个例子进行说明
3:实现原理:
切面的意义何在?
首先根据上例,假设我们实现了一个通用的权限检查模块,那么就可以在这层切面上进
行统一的集中式权限管理。权限检查模块可以和业务逻辑代码分离,而业务逻辑组件则无需关心权限方面的问题。系统大部分有权限检查模块,用的时候直接拿来用这个切面。也就是说,通过切面,我们可以将系统中各个不同层次上的问题隔离开来,实现统一集约式处理。各切面只需集中于自己领域内的逻辑实现。
这一方面使得开发逻辑更加清晰,专业化分工更加易于进行;另一方面,由于切面的隔
离,降低了耦合性,我们就可以在不同的应用中将各个切面组合使用,从而使得代码可重用
性大大增强。
如果是面向接口的动态代理的实现,即JDKProxy,其代理对象必须是某个接口的实现,使用java.lang.reflect.Proxy类根据一个被代理对象产生一个代理对象userDAOProxy,通过Proxy类的调用静态方法newProxyInstance,根据要实现的接口来产生(这里为UserDao接口)(也就是说接口里面有哪些方法,我生成的代理里面就有哪些方法);以及实现java.lang.reflect.InvocationHandler接口,实现invoke方法实现方法的截获处理,也就是在方法的前后加上业务逻辑。
当你想在多个方法前后加上业务逻辑的时候,可以使用动态代理,更加灵活方便,代码的可重用性大大的提高。
LogIntercepter li = new LogIntercepter();
li.setTarget(userDAO);//引入一个被代理的对象userDAO
UserDao userDAOProxy = (UserDao) Proxy.newProxyInstance(userDAO.getClass().getClassLoader(),userDAO.getClass().getInterfaces(), li);
public Object invoke(Object proxy, Method m, Object[] args)
throws Throwable {
beforeMethod();//在方法前面添加业务逻辑,也就是日志
m.invoke(target, args);//target Method方法所属的对象,表示被代理对象动态调用invoke()
return null;
}
如果没有实现接口,也没有关系,可以用CGLib(面向Class)实现AOP。
CGLibProxy与JDKProxy的代理机制基本类似,只是其动态代理的代理对象并非某个接口的实现,而是针对目标类扩展的子类。话句话说JDKProxy返回动态代理类,是目标类所实现接口的另一个实现版本,它实现了对目标类的代理(如同UserDAOProxy与UserDAOImp的关系),而CGLibProxy返回的动态代理类,则是目标代理类的一个子类(代理类扩展了UserDaoImpl类)
Enhancer和MethodInterceptor在CGLib中负责完成代理对象创建和方法截获处理。
Enhancer创建代理对象,实现MethodInterceptor接口,实现intercept方法来进行方法截取处理。
(CGLib (Code Generation Library) 字节码类库是一个强大的,高性能,高质量的Code生成类库。它可以在运行期扩展Java类与实现Java接口。Hibernate用它来实现PO字节码的动态生成。CGLib 比 Java 的 java.lang.reflect.Proxy 类更强的在于它不仅可以接管接口类的方法,还可以接管普通类的方法。)
JDK动态代理和CGLIB字节码生成的区别:
* JDK动态代理只能对实现了接口的类生成代理,而不能针对类.
* CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法
因为是继承,所以该类或方法最好不要声明成final。
JDK代理是不需要以来第三方的库,只要JDK环境就可以进行代理,它有几个要求
* 实现InvocationHandler
* 使用Proxy.newProxyInstance产生代理对象
* 被代理的对象必须要实现接口
CGLib 必须依赖于CGLib的类库,但是它需要类来实现任何接口代理的是指定的类生成一个子类,覆盖其中的方法,是一种继承
AOP的应用:
做权限的检查:设计完备的权限管理组件,完成以往需要大费周折才能完成的权限判定功能。
但是目前还没有一个完善的实现,一方面是因为权限检查过于复杂多变,不同的业务逻辑中的权限判定逻辑可能多种多样;另一方面,就目前的AOP应用的粒度而言,“权限管理”作为一个切面显得过于庞大,需要进一步设计,设计复杂,实现难度大。
,做日志,做审计,做性能,做事务的处理。项目里面主要用在了声明式的事务处理上。
1、JDK方式:PersonService为接口,PersonServiceBean为实现类,
public class JDKProxyFactory implements InvocationHandler {
private
Object targetObject;
public Object createProxyIntance(Object
targetObject)
{
this.targetObject=targetObject;
return
Proxy.newProxyInstance(this.targetObject.getClass().getClassLoader(),
this.targetObject.getClass().getInterfaces(), this);
}
public Object invoke(Object proxy, Method method, Object[] args)
throws
Throwable {
PersonServiceBean
person=(PersonServiceBean)this.targetObject;
Object result=null;
if(person.getUser()!=null)
{
result =
method.invoke(targetObject, args);
}
return result;
}
}
2、使用CGlib包实现:PersonServiceBean为实现类, 而没有PersonService接口,
public class CGlibProxyFactory implements MethodInterceptor{
private
Object targetObject;
public Object createProxyInstance(Object
targetObject)
{
this.targetObject=targetObject;
Enhancer
enhancer=new Enhancer();
enhancer.setSuperclass(this.targetObject.getClass());//设置目标类的子类,该子类会覆盖所有父类中的非final方法
enhancer.setCallback(this);//设置回调
return enhancer.create();
}
public Object intercept(Object proxy, Method method, Object[]
args,
MethodProxy methodProxy) throws Throwable {
PersonServiceBean
person=(PersonServiceBean)this.targetObject;
Object result=null;
if(person.getUser()!=null)
{
result =
methodProxy.invoke(targetObject, args);
}
return null;
}
}