一,代理模式
代理模式:
- 为对象提供一个替身,以控制对这个对象的访问。即通过代理对象访问目标对象,可以在目标对象的基础上实现扩展,增强额外的功能;
- 被代理对象可以是远程对象、创建开销大的对象或需要安全控制的对象等;
- 主要有:静态代理,jdk动态代理,Cglib动态代理
二,原理类图
意图: 为其他对象提供一种代理以控制对这个对象的访问。
适用性:
在需要用比较通用和复杂的对象指针代替简单的指针的时候,使用Proxy模式。下面是一 些可以使用Proxy 模式常见情况:
- 远程代理(Remote Proxy )为一个对象在不同的地址空间提供局部代表。 NEXTSTEP[Add94] 使用NXProxy 类实现了这一目的。Coplien[Cop92] 称这种代理为“大使” (Ambassador )。
- 虚代理(Virtual Proxy )根据需要创建开销很大的对象。在动机一节描述的ImageProxy 就是这样一种代理的例子。
- 保护代理(Protection Proxy )控制对原始对象的访问。保护代理用于对象应该有不同 的访问权限的时候。例如,在Choices 操作系统[ CIRM93]中KemelProxies为操作系统对象提供 了访问保护。
- 智能指引(Smart Reference )取代了简单的指针,它在访问对象时执行一些附加操作。 它的典型用途包括:对指向实际对象的引用计数,这样当该对象没有引用时,可以自动释放它(也称为SmartPointers[Ede92 ] )。
当第一次引用一个持久对象时,将它装入内存。
在访问一个实际对象前,检查是否已经锁定了它,以确保其他对象不能改变它。
三,实例
静态代理
静态代理: 定义接口或者父类,被代理对象和代理对象一起实现或继承;
优点: 在不修改被代理类的前提下实现功能扩展增强;
缺点: 同时需要继承或实现同一个接口,一旦接口方法增加所有代理和被代理类都需要调整;
接口
package com.neei.proxy.statis;
/**
* @param
* @Author: AaNeei
* @Date: 2019/10/11 21:35
* @Description: 游学网
* @throws:
*/
public interface ITeacher {
void teach();
}
被代理类
package com.neei.proxy.statis;
/**
* @param
* @Author: AaNeei
* @Date: 2019/10/11 21:34
* @Description: 游学网
* @throws:
*/
public class Teacher implements ITeacher {
@Override
public void teach(){
System.out.println("老师上课中。。。。");
}
}
代理类
package com.neei.proxy.statis;
/**
* @param
* @Author: AaNeei
* @Date: 2019/10/11 21:36
* @Description: 游学网
* @throws:
*/
public class TeacherProxy implements ITeacher {
private ITeacher iTeacher;
public TeacherProxy(ITeacher iTeacher) {
this.iTeacher = iTeacher;
}
@Override
public void teach() {
System.out.println("静态代理开始-----功能增强");
iTeacher.teach();
System.out.println("功能增强完毕-----静态代理结束");
}
}
调用
package com.neei.proxy.statis;
/**
* @param
* @Author: AaNeei
* @Date: 2019/10/11 21:38
* @Description: 游学网
* @throws:
*/
public class Client {
public static void main(String[] args) {
Teacher teacher = new Teacher();
TeacherProxy proxy=new TeacherProxy(teacher);
proxy.teach();
}
}
JDK动态代理
JDK动态代理:
- 代理对象不需要实现接口,被代理对象需要实现接口,又叫接口代理
- 代理对象的生成是通过JDK的API,动态的在内存中构建的;
接口
package com.neei.proxy.statis;
/**
* @param
* @Author: AaNeei
* @Date: 2019/10/11 21:35
* @Description: 游学网
* @throws:
*/
public interface ITeacher {
void teach();
}
被代理类
package com.neei.proxy.statis;
/**
* @param
* @Author: AaNeei
* @Date: 2019/10/11 21:34
* @Description: 游学网
* @throws:
*/
public class Teacher implements ITeacher {
@Override
public void teach(){
System.out.println("老师上课中。。。。");
}
}
代理类
package com.neei.proxy.dynamic;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
* @param
* @Author: AaNeei
* @Date: 2019/10/11 21:56
* @Description: 游学网
* @throws:
*/
public class ProxyFactory {
private Object object;
public ProxyFactory(Object object) {
this.object = object;
}
public Object getProxyInstance() {
return Proxy.newProxyInstance(
//代理对象的类加载器
object.getClass().getClassLoader(),
//代理对象实现的接口集合
object.getClass().getInterfaces(),
//方法处理拦截器
new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("JDK动态代理开始---");
return method.invoke(object, args);
}
});
}
}
调用
package com.neei.proxy.dynamic;
/**
* @param
* @Author: AaNeei
* @Date: 2019/10/11 21:55
* @Description: 游学网
* @throws:
*/
public class Client {
public static void main(String[] args) {
ITeacher teacher=new Teacher();
ITeacher instance = (ITeacher) new ProxyFactory(teacher).getProxyInstance();
instance.teach();
}
}
CGLIB动态代理
CGLIB动态代理:
- 静态代理和jdk动态代理都需要实现接口,如果没有接口实现的时候就只能使用cglib代理;
- cglib又叫子类代理,在内存中构建一个子类对象(不能是final类),实现对目标对象的功能扩展;
- cglib是个高性能的代码生成包,可以在运行期扩展java类实现、java接口等,多用于AOP框架,实现方法拦截;
- cglib底层采用了asm框架的字节码技术转换字节码并生成新的类;
被代理类
package com.neei.proxy.statis;
/**
* @param
* @Author: AaNeei
* @Date: 2019/10/11 21:34
* @Description: 游学网
* @throws:
*/
public class Teacher implements ITeacher {
@Override
public void teach(){
System.out.println("老师上课中。。。。");
}
}
代理类
package com.neei.proxy.dynamic;
import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;
/**
* @param
* @Author: AaNeei
* @Date: 2019/10/11 21:56
* @Description: 游学网
* @throws:
*/
public class ProxyFactory implements MethodInterceptor {
private Object object;
public ProxyFactory(Object object) {
this.object = object;
}
public Object getProxyInstance() {
//创建工具类
Enhancer enhancer = new Enhancer();
//设置目标类
enhancer.setSuperclass(object.getClass());
//回调目标类方法
enhancer.setCallback(this);
//返回代理对象
return enhancer.create();
}
@Override
public Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
System.out.println("cglib动态代理开始----");
Object invoke = method.invoke(object, args);
return invoke;
}
}
调用
package com.neei.proxy.dynamic;
/**
* @param
* @Author: AaNeei
* @Date: 2019/10/11 21:55
* @Description: 游学网
* @throws:
*/
public class Client {
public static void main(String[] args) {
Teacher teacher=new Teacher();
Teacher instance = (Teacher) new ProxyFactory(teacher).getProxyInstance();
instance.teach();
}
}
四,源码分析
JDK源码中使用的代理模式,如Proxy.newProxyInstance;