什么是代理
代理:自己不做,招人帮忙
代理模式:在一个原有功能的基础上添加新的功能;
分类:静态代理,动态代理
静态代理
原有模式
public class TeamService {
public void add(){
System.out.println("开始事务");
System.out.println("核心业务");
System.out.println("提交事务");
System.out.println("回滚事务");
}
}
基于类的静态代理
将服务性代码分离出来,只保留核心业务
public class TeamService {
public void add(){
System.out.println("核心业务");
}
}
public class staticProxy extends TeamService{
/*
*基于类的静态代理
* 要求继承被代理的类
* 弊端,每次只能代理一个类
* */
public void add(){
System.out.println("开始事务");
super.add();//核心业务由被代理对象完成,其他服务型业务由代理类完成
System.out.println("提交事务");
System.out.println("回滚事务");
}
}
基于接口的静态代理
为核心业务创建一个接口
public interface TeamService {
public void add();
}
定义两个代理类
测试
提取出切面代码,作为AOP接口
共有四个位置可以将切面代码编织到核心业务代码中
两个实现类
测试
静态代理总结
- 可以做到不修改目标对象的功能前提下,对目标对象功能进行扩展
- 因为代理对象需要与目标对象实现一样的接口。所以会有很多代理类,类太多;
- 一旦接口增加方法,目标对象和代理对象都要维护。
动态代理
程序运行的时候,根据要被代理的对象动态生成代理类
分类:
- JDK动态代理
- CGLIB动态代理
基于JDK的动态代理---创建代理对象的接口
newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
ClassLoader:类加载器,因为动态代理类,借助别人的类加载器。一般使用被代理对象的类加载器
Class<?>[] interfaces:接口类对象的集合,针对接口的代理,针对那个接口做代理,一般使用的就是被代理对象的接口;
InvocationHandler:句柄,回调函数,编写代理规则代码
直接编写测试类
结构化设计
方式一
测试
方式二
测试
总结,代理对象不需要实现接口,但是目标对象一定要实现接口;否则不能用JDK动态代理
基于CGLIB的动态代理
CGLib代理,又叫做子类代理,在内存中构建一个子类对象从而实现对目标对象功能的扩展。
- JDK的动态代理有一个限制,就是使用动态代理 的对象必须实现一个或多个接口,如果想代理没有实现接口的类,就可以使用CGLib。
- CGLIB是一个强大的高性能的代码生成包,它可以在运行期扩展Java类与实现Java接口。它广泛的被许多AOP的框架使用,例如Spring AOP和dynaop,为他们提供方法的interception。
- CGLIB包的底层是通过使用一个小而快的字节码处理框架ASM,来转换字节码并生成新的类。不鼓励直接使用ASM,因为它要求你必须对JVM内部结构包括class文件的格式和指令集都很熟悉。
直接编写测试类
结构化设计