什么是代理模式
代理模式是指给某一个对象提供一个代理或占位符,并由代理对象来控制对原对象的访问。代理对象在客户端和目标对象之间起到中介的作用。
代理模式包含以下几个角色:
Subject(抽象主题角色):声明了真实主题和代理主题的共同接口,这样在任何使用真实主题的地方都可以使用代理主题。客户端通常需要针对抽象主题角色编程。
Proxy(代理主题角色):包含了对真实主题的引用,从而可以在任何时候操作真实主题对象。在代理主题角色中提供一个与真实主题角色相同的接口,以便在任何时候都可以替代真实主题。代理主题角色还可以控制对真实主题的使用,负责在需要的时候创建和删除真实主题,并对真实主题对象的使用加以限制。
RealSubject(真实主题角色):定义了代理角色所代表的真实对象,在真实主题角色中实现了真实的业务操作。
java中代理分为静态代理和动态代理:
静态代理:代理对象和被代理对象实现同一个接口,代理类在编译器就生成。
动态代理:在java运行时动态生成。
代理模式的优缺点
优点
- 降低了系统耦合。
- 客户端可以针对抽象主题角色编程,增加或者替换代理类无需修改源代码,符合开闭原则。
- 静态代理实现简单。
缺点
- 代理类和被代理类必须实现同一个接口。
- 静态代理中,接口发生修改,代理类也要进行修改。
代理模式的应用场景
- 隐藏某个类或者类中的某些方法。
代理模式的案例
// 1. 静态代理
// 抽象主题角色
public interface ITeacherDao {
void teach();
}
// 真实主题角色
public class TeacherDao implements ITeacherDao {
@Override
public void teach() {
System.out.println(" 老师授课中 。。。。。");
}
}
// 代理主题角色
public class TeacherDaoProxy implements ITeacherDao {
private ITeacherDao target;
public TeacherDaoProxy(ITeacherDao target) {
this.target = target;
}
@Override
public void teach() {
System.out.println("开始代理 完成某些操作。。。。。 ");
target.teach();
System.out.println("提交。。。。。");
}
}
public static void main(String[] args) {
//创建目标对象(被代理对象)
TeacherDao teacherDao = new TeacherDao();
//创建代理对象, 同时将被代理对象传递给代理对象
TeacherDaoProxy teacherDaoProxy = new TeacherDaoProxy(teacherDao);
//通过代理对象,调用到被代理对象的方法
//即:执行的是代理对象的方法,代理对象再去调用目标对象的方法
teacherDaoProxy.teach();
}
/******************************************************************/
// 2. JDK动态代理
// 抽象主题角色
public interface ITeacherDao {
void teach();
void sayHello(String name);
}
// 真实主题角色
public class TeacherDao implements ITeacherDao {
@Override
public void teach() {
System.out.println(" 老师授课中.... ");
}
@Override
public void sayHello(String name) {
System.out.println("hello " + name);
}
}
// 代理主题角色
public class ProxyFactory {
private Object target;
public ProxyFactory(Object target) {
this.target = target;
}
public Object getProxyInstance() {
//说明
/*
* public static Object newProxyInstance(ClassLoader loader,
Class<?>[] interfaces,
InvocationHandler h)
//1. ClassLoader loader : 指定当前目标对象使用的类加载器, 获取加载器的方法固定
//2. Class<?>[] interfaces: 目标对象实现的接口类型,使用泛型方法确认类型
//3. InvocationHandler h : 事情处理,执行目标对象的方法时,会触发事情处理器方法, 会把当前执行的目标对象方法作为参数传入
*/
return Proxy.newProxyInstance(target.getClass().getClassLoader(),
target.getClass().getInterfaces(),
(proxy, method, args) -> {
System.out.println("JDK代理开始~~");
//反射机制调用目标对象的方法
Object returnVal = method.invoke(target, args);
System.out.println("JDK代理提交");
return returnVal;
});
}
}
public static void main(String[] args) {
//创建目标对象
ITeacherDao target = new TeacherDao();
//给目标对象,创建代理对象, 可以转成 ITeacherDao
ITeacherDao proxyInstance = (ITeacherDao) new ProxyFactory(target).getProxyInstance();
// proxyInstance=class com.sun.proxy.$Proxy0 内存中动态生成了代理对象
System.out.println("proxyInstance=" + proxyInstance.getClass());
//通过代理对象,调用目标对象的方法
proxyInstance.teach();
proxyInstance.sayHello(" tom ");
}
代理模式在源码中的应用
DefaultAopProxyFactory
// 在AOP代理类中根据代理对象是否实现接口判断是jdk动态代理还是cglib代理
public class DefaultAopProxyFactory implements AopProxyFactory, Serializable {
private static final long serialVersionUID = 7930414337282325166L;
@Override
public AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException {
if (!NativeDetector.inNativeImage() &&
(config.isOptimize() || config.isProxyTargetClass() || hasNoUserSuppliedProxyInterfaces(config))) {
Class<?> targetClass = config.getTargetClass();
if (targetClass == null) {
throw new AopConfigException("TargetSource cannot determine target class: " +
"Either an interface or a target is required for proxy creation.");
}
if (targetClass.isInterface() || Proxy.isProxyClass(targetClass) || ClassUtils.isLambdaClass(targetClass)) {
return new JdkDynamicAopProxy(config);
}
return new ObjenesisCglibAopProxy(config);
}
else {
return new JdkDynamicAopProxy(config);
}
}
......
}