一. 概述
代理(Proxy)是一种设计模式,提供对目标对象的额外访问方式。
通过代理,就可以在目标对象方法的基础上添加新的方法。
二. 静态代理
逻辑
代理对象和目标对象同时继承父类或调用接口
代理对象生成时传入对象参数
代理对象在自身的接口方法重写中,调用目标对象方法,织入增强语句
范例
/**
* 接口
*/
public interface IUserDao {
void save();
}
/**
* 接口实现
* 目标对象
*/
public class UserDao implements IUserDao {
public void save() {
System.out.println("----已经保存数据!----");
}
}
/**
* 代理对象,静态代理
*/
public class UserDaoProxy implements IUserDao{
//接收保存目标对象
private IUserDao target;
public UserDaoProxy(IUserDao target){
this.target=target;
}
public void save() {
System.out.println("开始事务...");
target.save();//执行目标对象的方法
System.out.println("提交事务...");
}
}
/**
* 测试类
*/
public class App {
public static void main(String[] args) {
//目标对象
UserDao target = new UserDao();
//代理对象,把目标对象传给代理对象,建立代理关系
UserDaoProxy proxy = new UserDaoProxy(target);
proxy.save();//执行的是代理的方法
}
}
总结
**优点:**可以在不修改目标对象的情况下,增加扩展功能
**缺点:**代理对象需要和目标对象实现相同接口,接口修改,二者都要修改
三. 动态代理
概念
与静态代理不同的是,代理对象不需要实现接口
代理对象的生成是使用 JDK 的 API ,在内存中动态构建(需要指定对象实现的接口类型)
也叫 JDK 代理,接口代理
**API **
static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,InvocationHandler h )
// ClassLoader :指定目标对象使用的类加载器
// Class<?>[] interfaces :目标对象实现的接口的类型
// InvocationHandler h :当前执行目标对象的方法(执行目标对象方法是,会出发事件处理器方法导入)
范例
/**
* 创建动态代理对象
* 动态代理不需要实现接口,但是需要指定接口类型
*/
public class ProxyFactory{
//维护一个目标对象
private Object target;
public ProxyFactory(Object target){
this.target=target;
}
//给目标对象生成代理对象
public Object getProxyInstance(){
return Proxy.newProxyInstance(
target.getClass().getClassLoader(),
target.getClass().getInterfaces(),
new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("开始事务2");
//执行目标对象方法
Object returnValue = method.invoke(target, args);
System.out.println("提交事务2");
return returnValue;
}
}
);
}
}
/**
* 测试类
*/
public class App {
public static void main(String[] args) {
// 目标对象
IUserDao target = new UserDao();
// 【原始的类型 class cn.itcast.b_dynamic.UserDao】
System.out.println(target.getClass());
// 给目标对象,创建代理对象
IUserDao proxy = (IUserDao) new ProxyFactory(target).getProxyInstance();
// class $Proxy0 内存中动态生成的代理对象
System.out.println(proxy.getClass());
// 执行方法 【代理对象】
proxy.save();
}
}
五. Cglib 代理
概念
在内存中创建一个子类对象实现对目标对象的功能拓展
也叫子类代理
目标对象可以没有实现接口(JDK 代理必须实现接口)
被运用在 Spring AOP 中,提供方法的 Interceptor(拦截)
代理类不能为 final
目标对象的方法为 final / static 是,不会被拦截,不会执行目标对象的其他方法
范例
/**
* 目标对象,没有实现任何接口
*/
public class UserDao {
public void save() {
System.out.println("----已经保存数据!----");
}
}
/**
* Cglib子类代理工厂
* 对UserDao在内存中动态构建一个子类对象
*/
public class ProxyFactory implements MethodInterceptor{
//维护目标对象
private Object target;
public ProxyFactory(Object target) {
this.target = target;
}
//给目标对象创建一个代理对象
public Object getProxyInstance(){
//1.工具类
Enhancer en = new Enhancer();
//2.设置父类
en.setSuperclass(target.getClass());
//3.设置回调函数
en.setCallback(this);
//4.创建子类(代理对象)
return en.create();
}
@Override
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
System.out.println("开始事务...");
//执行目标对象的方法
Object returnValue = method.invoke(target, args);
System.out.println("提交事务...");
return returnValue;
}
}
/**
* 测试类
*/
public class App {
@Test
public void test(){
//目标对象
UserDao target = new UserDao();
//代理对象
UserDao proxy = (UserDao)new ProxyFactory(target).getProxyInstance();
//执行代理对象的方法
proxy.save();
}
}
总结
Spring AOP 中,若目标对象需要实现接口。使用 JDK 代理,不然,使用 Cglib 代理