一、什么是代理模式?
定义: 为其他对象提供一种代理以控制对这个对象的访问,重点在控制以及扩展已有类的功能
通俗理解: 就是一个中介,通过该对象来访问另外一个对象
编程思想: 不要随意去修改别人已经写好的代码或者方法,如果需改修改,可以通过代理的方式来扩展 该方法
案例理解: 假设我们想邀请一位明星,那么并不是直接连接明星,而是联系明星的经纪人,来达到同样的目的.明星就是一个目标对象,他只要负责活动中的节目,而其他琐碎的事情就交给他的代理人(经纪人)来解决。
代理模式理解重点:代理对象与目标对象.代理对象是对目标对象的扩展,并会调用目标对象
二、代理模式种类?
代理模式分为 静态代理模式,动态代理模式, cglib代理,以下分别说明这三种代理模式。
三、静态代理模式?
特点: 需要定义接口或者父类,被代理对象与代理对象一起实现相同的接口或者是继承相同父类.
角色?
目标接口: 定义需要代理的对象接口
目标对象: 定义需要代理的对象,实现目标接口
代理对象: 定义代理对象,实现目标接口,持有目标对象引用
分析?
优点: 可以做到在不修改目标对象的功能前提下, 使用代理对象实现对目标功能扩展.
缺点: 因为代理对象需要与目标对象实现一样的接口,所以会有很多代理类,类太多.同时如果对目标对象执行方法前后的操作都一样,每个方法中存在前后执行代码的冗余代码。一旦接口增 加方法,目标对象与代理对象都要维护.
如何解决静态代理中的缺点呢?
答案是可以使用动态代理方式
四、动态代理模式?
由来: 解决使用静态代理造成代理类太多,当接口修改目标与代理对象都要维护。
特点: 1.代理对象,不需要实现接口
2.代理对象的生成,是利用JDK的API, 动态的在内存中构建代理对象(需要我们指定创建代理对象/
目标对象实现的接口的类型)
3. 动态代理也叫做:JDK代理,接口代理
4. 代理对象不需要实现接口,但是目标对象一定要实现接口,否则不能用动态代理
使用: JDK中生成代理对象的API
代理类所在包:java.lang.reflect.Proxy
JDK实现代理只需要使用newProxyInstance方法,但是该方法需要接收三个参数,完整的写法是:
static Object newProxyInstance(ClassLoader loader, Class[] interfaces,InvocationHandler h )
注意该方法是在Proxy类中是静态方法,且接收的三个参数依次为:
ClassLoader loader,:指定当前目标对象使用类加载器,获取加载器的方法是固定的
Class[] interfaces,:目标对象实现的接口的类型,使用泛型方式确认类型
InvocationHandler h:事件处理,执行目标对象的方法时,会触发事件处理器的方法,会把当前执行目标对象的方法作为参数传入
五、Cglib代理
问题? 如果一个类没有实现接口,那怎么为这个类创建代理对象呢?这时候就需要使用到Cglib代理了。
特点: 目标对象不需要实现接口
使用: 1、下载Cglib jar包
2、给目标对象创建代理对象
public class ProxyFactoryimplements MethodInterceptor {
private Objecttarget;
public ProxyFactory(Object target) {
this.target = target;
}
// 使用cglib给目标对象创建代理对象
public ObjectgetProxyInstance() {
// 1、创建工具类
Enhancer enhancer =new Enhancer();
// 2、设置父类
enhancer.setSuperclass(target.getClass());
// 3、设置回调
enhancer.setCallback(this);
// 4、创建子类
return enhancer.create();
}
@Override
public Objectintercept(Object o, Method method, Object[] objects, MethodProxy methodProxy)throws Throwable {
android.util.Log.i("zqr", "on pre save user");
Object object = method.invoke(target, objects);
android.util.Log.i("zqr", "on After save user");
return object;
}
}
三、代理模式与适配器模式,装饰者模式区别?
1、和适配器模式的区别:适配器模式主要改变所考虑对象的接口,而代理模式不能改变所代理类的接口。